mirror of
https://github.com/NikitaIvanovV/ctpv.git
synced 2025-01-22 07:58:34 +01:00
327 lines
6.5 KiB
C
327 lines
6.5 KiB
C
#include <stdio.h>
|
|
#include <magic.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <signal.h>
|
|
#include <unistd.h>
|
|
#include <sys/stat.h>
|
|
#include <openssl/md5.h>
|
|
|
|
#include "error.h"
|
|
#include "server.h"
|
|
#include "preview.h"
|
|
#include "previews.h"
|
|
|
|
#define ANY_TYPE "*"
|
|
|
|
static const char any_type[] = ANY_TYPE;
|
|
|
|
static magic_t magic;
|
|
|
|
static struct {
|
|
enum {
|
|
MODE_PREVIEW,
|
|
MODE_SERVER,
|
|
MODE_CLEAR,
|
|
MODE_END,
|
|
MODE_LIST,
|
|
MODE_MIME,
|
|
} mode;
|
|
char *server_id_s;
|
|
} ctpv = { .mode = MODE_PREVIEW };
|
|
|
|
static void cleanup(void) {
|
|
cleanup_previews();
|
|
if (magic != NULL)
|
|
magic_close(magic);
|
|
}
|
|
|
|
static int init_magic()
|
|
{
|
|
ERRCHK_RET(!(magic = magic_open(MAGIC_MIME_TYPE)), FUNCFAILED("magic_open"),
|
|
magic_error(magic));
|
|
|
|
ERRCHK_RET(magic_load(magic, NULL) != 0, FUNCFAILED("magic_load"),
|
|
magic_error(magic));
|
|
|
|
return OK;
|
|
}
|
|
|
|
static void init_previews_v(void)
|
|
{
|
|
init_previews(previews, LEN(previews));
|
|
}
|
|
|
|
static const char *get_mimetype(char const *path)
|
|
{
|
|
const char *r = magic_file(magic, path);
|
|
if (!r) {
|
|
PRINTINTERR(FUNCFAILED("magic_file"), magic_error(magic));
|
|
return NULL;
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
static const char *get_ext(char const *path)
|
|
{
|
|
const char *base;
|
|
|
|
if ((base = strrchr(path, '/')))
|
|
base += sizeof(*base);
|
|
else
|
|
base = path;
|
|
|
|
const char *dot = strchr(base, '.');
|
|
if (!dot || dot == base)
|
|
return NULL;
|
|
|
|
return &dot[1];
|
|
}
|
|
|
|
static int check_file(char const *f)
|
|
{
|
|
if (!f) {
|
|
print_error("file not given");
|
|
return ERR;
|
|
}
|
|
|
|
if (access(f, R_OK) != 0) {
|
|
print_errorf("failed to access '%s': %s", f, ERRNOS);
|
|
return ERR;
|
|
}
|
|
|
|
return OK;
|
|
}
|
|
|
|
static int is_newer(int *resp, char *f1, char *f2)
|
|
{
|
|
struct stat stat1, stat2;
|
|
ERRCHK_RET(stat(f1, &stat1) == -1, FUNCFAILED("stat"), ERRNOS);
|
|
ERRCHK_RET(stat(f2, &stat2) == -1, FUNCFAILED("stat"), ERRNOS);
|
|
|
|
int sec_d = stat1.st_mtim.tv_sec - stat2.st_mtim.tv_sec;
|
|
if (sec_d < 0)
|
|
goto older;
|
|
else if (sec_d == 0 && stat1.st_mtim.tv_nsec <= stat2.st_mtim.tv_nsec)
|
|
goto older;
|
|
|
|
*resp = 1;
|
|
return OK;
|
|
|
|
older:
|
|
*resp = 0;
|
|
return OK;
|
|
}
|
|
|
|
static void md5_string(char *buf, size_t len, char *s)
|
|
{
|
|
unsigned char out[MD5_DIGEST_LENGTH];
|
|
char b[16];
|
|
|
|
MD5((const unsigned char *)s, strlen(s), out);
|
|
|
|
buf[0] = '\0';
|
|
for(unsigned int i = 0; i < LEN(out); i++) {
|
|
snprintf(b, LEN(b)-1, "%02x", out[i]);
|
|
strncat(buf, b, len);
|
|
}
|
|
}
|
|
|
|
static int get_cache_file(char *buf, size_t len, char *file)
|
|
{
|
|
ERRCHK_RET_OK(get_cache_dir(buf, len, "ctpv/"));
|
|
|
|
{
|
|
char dir[len];
|
|
strncpy(dir, buf, LEN(dir) - 1);
|
|
ERRCHK_RET(mkpath(dir, 0700) == -1, FUNCFAILED("mkpath"), ERRNOS);
|
|
}
|
|
|
|
char name[64];
|
|
md5_string(name, LEN(name) - 1, file);
|
|
strncat(buf, name, len - 1);
|
|
|
|
return OK;
|
|
}
|
|
|
|
static int check_cache(int *resp, char *file, char *cache_file)
|
|
{
|
|
if (access(cache_file, F_OK) != 0) {
|
|
*resp = 0;
|
|
return OK;
|
|
}
|
|
|
|
return is_newer(resp, cache_file, file);
|
|
}
|
|
|
|
#define GET_PARG(a, i) (a) = argc > (i) ? argv[i] : NULL
|
|
|
|
static int preview(int argc, char *argv[])
|
|
{
|
|
char *f, *w, *h, *x, *y, *id;
|
|
GET_PARG(f, 0);
|
|
GET_PARG(w, 1);
|
|
GET_PARG(h, 2);
|
|
GET_PARG(x, 3);
|
|
GET_PARG(y, 4);
|
|
GET_PARG(id, 5);
|
|
|
|
ERRCHK_RET_OK(check_file(f));
|
|
|
|
ERRCHK_RET_OK(init_magic());
|
|
|
|
init_previews_v();
|
|
|
|
const char *mimetype;
|
|
ERRCHK_RET(!(mimetype = get_mimetype(f)));
|
|
|
|
char cache_file[FILENAME_MAX];
|
|
ERRCHK_RET_OK(get_cache_file(cache_file, LEN(cache_file), f));
|
|
|
|
int cache_valid;
|
|
ERRCHK_RET_OK(check_cache(&cache_valid, f, cache_file));
|
|
|
|
PreviewArgs args = {
|
|
.f = f, .w = w, .h = h, .x = x, .y = y, .id = id,
|
|
.cache_file = cache_file, .cache_valid = cache_valid,
|
|
};
|
|
|
|
return run_preview(get_ext(f), mimetype, &args);
|
|
}
|
|
|
|
static int server(void)
|
|
{
|
|
return server_listen(ctpv.server_id_s);
|
|
}
|
|
|
|
static int clear(void)
|
|
{
|
|
return server_clear(ctpv.server_id_s);
|
|
}
|
|
|
|
static int end(void)
|
|
{
|
|
return server_end(ctpv.server_id_s);
|
|
}
|
|
|
|
static int list(void)
|
|
{
|
|
init_previews_v();
|
|
|
|
size_t len;
|
|
Preview p, **list = get_previews_list(&len);
|
|
const char *e, *t, *s;
|
|
|
|
puts("List of available previews:");
|
|
|
|
for (size_t i = 0; i < len; i++) {
|
|
p = *list[i];
|
|
e = p.ext;
|
|
t = p.type;
|
|
s = p.subtype;
|
|
|
|
if (!e)
|
|
e = any_type;
|
|
|
|
if (!t) {
|
|
t = any_type;
|
|
s = any_type;
|
|
} else if (!s) {
|
|
s = any_type;
|
|
}
|
|
|
|
printf("\t%-15s .%-6s %s/%s\n", p.name, e, t, s);
|
|
}
|
|
|
|
puts("\nNote: '" ANY_TYPE "' means that it matches any.");
|
|
return OK;
|
|
}
|
|
|
|
static int mime(int argc, char *argv[])
|
|
{
|
|
char const *f, *mimetype;
|
|
|
|
for (int i = 0; i < argc; i++) {
|
|
f = argv[i];
|
|
ERRCHK_RET_OK(check_file(f));
|
|
|
|
ERRCHK_RET_OK(init_magic());
|
|
|
|
mimetype = get_mimetype(f);
|
|
ERRCHK_RET(!mimetype);
|
|
|
|
if (argc > 1)
|
|
printf("%s: ", f);
|
|
|
|
printf(".%s ", get_ext(f));
|
|
puts(mimetype);
|
|
}
|
|
|
|
return OK;
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
program = argc > 0 ? argv[0] : "ctpv";
|
|
|
|
int c;
|
|
while ((c = getopt(argc, argv, "s:c:e:lm")) != -1) {
|
|
switch (c) {
|
|
case 's':
|
|
ctpv.mode = MODE_SERVER;
|
|
ctpv.server_id_s = optarg;
|
|
break;
|
|
case 'c':
|
|
ctpv.mode = MODE_CLEAR;
|
|
ctpv.server_id_s = optarg;
|
|
break;
|
|
case 'e':
|
|
ctpv.mode = MODE_END;
|
|
ctpv.server_id_s = optarg;
|
|
break;
|
|
case 'l':
|
|
ctpv.mode = MODE_LIST;
|
|
break;
|
|
case 'm':
|
|
ctpv.mode = MODE_MIME;
|
|
break;
|
|
default:
|
|
return EXIT_FAILURE;
|
|
}
|
|
}
|
|
|
|
argc -= optind;
|
|
argv = &argv[optind];
|
|
|
|
int ret;
|
|
switch (ctpv.mode) {
|
|
case MODE_PREVIEW:
|
|
ret = preview(argc, argv);
|
|
break;
|
|
case MODE_SERVER:
|
|
ret = server();
|
|
break;
|
|
case MODE_CLEAR:
|
|
ret = clear();
|
|
break;
|
|
case MODE_END:
|
|
ret = end();
|
|
break;
|
|
case MODE_LIST:
|
|
ret = list();
|
|
break;
|
|
case MODE_MIME:
|
|
ret = mime(argc, argv);
|
|
break;
|
|
default:
|
|
PRINTINTERR("unknowm mode: %d", ctpv.mode);
|
|
ret = ERR;
|
|
break;
|
|
}
|
|
|
|
cleanup();
|
|
|
|
return ret == OK ? EXIT_SUCCESS : EXIT_FAILURE;
|
|
}
|