293 lines
8.0 KiB
C
293 lines
8.0 KiB
C
/* print.c - Alpine Package Keeper (APK)
|
|
*
|
|
* Copyright (C) 2005-2008 Natanael Copa <n@tanael.org>
|
|
* Copyright (C) 2008-2011 Timo Teräs <timo.teras@iki.fi>
|
|
* All rights reserved.
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0-only
|
|
*/
|
|
|
|
#include <assert.h>
|
|
#include <stdio.h>
|
|
#include <stdarg.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/types.h>
|
|
|
|
#include "apk_defines.h"
|
|
#include "apk_print.h"
|
|
#include "apk_io.h"
|
|
|
|
const char *apk_error_str(int error)
|
|
{
|
|
if (error < 0)
|
|
error = -error;
|
|
switch (error) {
|
|
case ECONNABORTED: return "network connection aborted";
|
|
case ECONNREFUSED: return "could not connect to server (check repositories file)";
|
|
case ENETUNREACH: return "network error (check Internet connection and firewall)";
|
|
case EAGAIN: return "temporary error (try again later)";
|
|
case APKE_EOF: return "unexpected end of file";
|
|
case APKE_DNS: return "DNS error (try again later)";
|
|
case APKE_URL_FORMAT: return "invalid URL (check your repositories file)";
|
|
case APKE_CRYPTO_ERROR: return "crypto error";
|
|
case APKE_CRYPTO_NOT_SUPPORTED: return "cryptographic algorithm not supported";
|
|
case APKE_CRYPTO_KEY_FORMAT: return "cryptographic key format not recognized";
|
|
case APKE_SIGNATURE_FAIL: return "signing failure";
|
|
case APKE_SIGNATURE_UNTRUSTED: return "UNTRUSTED signature";
|
|
case APKE_SIGNATURE_INVALID: return "BAD signature";
|
|
case APKE_FORMAT_NOT_SUPPORTED: return "file format not supported (in this applet)";
|
|
case APKE_PKGVERSION_FORMAT: return "package version is invalid";
|
|
case APKE_DEPENDENCY_FORMAT: return "dependency format is invalid";
|
|
case APKE_ADB_COMPRESSION: return "ADB compression not supported";
|
|
case APKE_ADB_HEADER: return "ADB header error";
|
|
case APKE_ADB_VERSION: return "incompatible ADB version";
|
|
case APKE_ADB_SCHEMA: return "ADB schema error";
|
|
case APKE_ADB_BLOCK: return "ADB block error";
|
|
case APKE_ADB_SIGNATURE: return "ADB signature block error";
|
|
case APKE_ADB_NO_FROMSTRING: return "ADB schema error (no fromstring)";
|
|
case APKE_ADB_LIMIT: return "ADB schema limit reached";
|
|
case APKE_ADB_PACKAGE_FORMAT: return "ADB package format";
|
|
case APKE_V2DB_FORMAT: return "v2 database format error";
|
|
case APKE_V2PKG_FORMAT: return "v2 package format error";
|
|
case APKE_V2PKG_INTEGRITY: return "v2 package integrity error";
|
|
case APKE_V2NDX_FORMAT: return "v2 index format error";
|
|
case APKE_PACKAGE_NOT_FOUND: return "could not find a repo which provides this package (check repositories file and run 'apk update')";
|
|
case APKE_INDEX_STALE: return "package mentioned in index not found (try 'apk update')";
|
|
case APKE_FILE_INTEGRITY: return "file integrity error";
|
|
case APKE_CACHE_NOT_AVAILABLE: return "cache not available";
|
|
case APKE_UVOL_NOT_AVAILABLE: return "uvol manager not available";
|
|
case APKE_UVOL_ERROR: return "uvol error";
|
|
case APKE_UVOL_ROOT: return "uvol not supported with --root";
|
|
case APKE_REMOTE_IO: return "remote server returned error (try 'apk update')";
|
|
default:
|
|
return strerror(error);
|
|
}
|
|
}
|
|
|
|
static const char *size_units[] = {"B", "KiB", "MiB", "GiB", "TiB"};
|
|
|
|
int apk_get_human_size_unit(apk_blob_t b)
|
|
{
|
|
for (int i = 0, s = 1; i < ARRAY_SIZE(size_units); i++, s *= 1024)
|
|
if (apk_blob_compare(b, APK_BLOB_STR(size_units[i])) == 0)
|
|
return s;
|
|
return 1;
|
|
}
|
|
|
|
const char *apk_get_human_size(off_t size, off_t *dest)
|
|
{
|
|
size_t i;
|
|
off_t s;
|
|
|
|
assert(size >= 0);
|
|
|
|
for (i = 0, s = size; s >= 10000 && i < ARRAY_SIZE(size_units); i++)
|
|
s /= 1024;
|
|
|
|
if (dest) *dest = s;
|
|
return size_units[min(i, ARRAY_SIZE(size_units) - 1)];
|
|
}
|
|
|
|
void apk_url_parse(struct apk_url_print *urlp, const char *url)
|
|
{
|
|
const char *authority, *path_or_host, *pw;
|
|
|
|
*urlp = (struct apk_url_print) {
|
|
.url = "",
|
|
.pwmask = "",
|
|
.url_or_host = url,
|
|
};
|
|
|
|
if (!(authority = strstr(url, "://"))) return;
|
|
authority += 3;
|
|
path_or_host = strpbrk(authority, "/@");
|
|
if (!path_or_host || *path_or_host == '/') return;
|
|
pw = strpbrk(authority, "@:");
|
|
if (!pw || *pw == '@') return;
|
|
*urlp = (struct apk_url_print) {
|
|
.url = url,
|
|
.pwmask = "*",
|
|
.url_or_host = path_or_host,
|
|
.len_before_pw = pw - url + 1,
|
|
};
|
|
}
|
|
|
|
void apk_out_reset(struct apk_out *out)
|
|
{
|
|
out->width = 0;
|
|
out->last_change++;
|
|
}
|
|
|
|
static int apk_out_get_width(struct apk_out *out)
|
|
{
|
|
struct winsize w;
|
|
|
|
if (out->width == 0) {
|
|
out->width = 50;
|
|
if (ioctl(fileno(out->out), TIOCGWINSZ, &w) == 0 &&
|
|
w.ws_col > 25)
|
|
out->width = w.ws_col;
|
|
}
|
|
|
|
return out->width;
|
|
}
|
|
|
|
static void log_internal(FILE *dest, const char *prefix, const char *format, va_list va)
|
|
{
|
|
if (dest != stdout) fflush(stdout);
|
|
if (prefix != NULL && prefix != APK_OUT_LOG_ONLY) fprintf(dest, "%s", prefix);
|
|
vfprintf(dest, format, va);
|
|
fprintf(dest, "\n");
|
|
fflush(dest);
|
|
}
|
|
|
|
void apk_out_fmt(struct apk_out *out, const char *prefix, const char *format, ...)
|
|
{
|
|
va_list va;
|
|
if (prefix != APK_OUT_LOG_ONLY) {
|
|
va_start(va, format);
|
|
log_internal(prefix ? out->err : out->out, prefix, format, va);
|
|
out->last_change++;
|
|
va_end(va);
|
|
}
|
|
|
|
if (out->log) {
|
|
va_start(va, format);
|
|
log_internal(out->log, prefix, format, va);
|
|
va_end(va);
|
|
}
|
|
}
|
|
|
|
void apk_out_log_argv(struct apk_out *out, char **argv)
|
|
{
|
|
char when[32];
|
|
struct tm tm;
|
|
time_t now = time(NULL);
|
|
|
|
if (!out->log) return;
|
|
fprintf(out->log, "\nRunning `");
|
|
for (int i = 0; argv[i]; ++i) {
|
|
fprintf(out->log, "%s%s", argv[i], argv[i+1] ? " " : "");
|
|
}
|
|
|
|
gmtime_r(&now, &tm);
|
|
strftime(when, sizeof(when), "%Y-%m-%d %H:%M:%S", &tm);
|
|
fprintf(out->log, "` at %s\n", when);
|
|
}
|
|
|
|
void apk_print_progress(struct apk_progress *p, size_t done, size_t total)
|
|
{
|
|
int bar_width;
|
|
int bar = 0;
|
|
char buf[64]; /* enough for petabytes... */
|
|
int i, percent = 0;
|
|
FILE *out;
|
|
|
|
if (p->last_done == done && (!p->out || p->last_out_change == p->out->last_change)) return;
|
|
if (p->fd != 0) {
|
|
i = snprintf(buf, sizeof(buf), "%zu/%zu\n", done, total);
|
|
if (apk_write_fully(p->fd, buf, i) != i) {
|
|
close(p->fd);
|
|
p->fd = 0;
|
|
}
|
|
}
|
|
p->last_done = done;
|
|
|
|
if (!p->out) return;
|
|
out = p->out->out;
|
|
if (!out) return;
|
|
|
|
bar_width = apk_out_get_width(p->out) - 6;
|
|
if (total > 0) {
|
|
bar = muldiv(bar_width, done, total);
|
|
percent = muldiv(100, done, total);
|
|
}
|
|
|
|
if (bar == p->last_bar && percent == p->last_percent && p->last_out_change == p->out->last_change)
|
|
return;
|
|
|
|
p->last_bar = bar;
|
|
p->last_percent = percent;
|
|
p->last_out_change = p->out->last_change;
|
|
|
|
fprintf(out, "\e7%3i%% ", percent);
|
|
|
|
for (i = 0; i < bar; i++)
|
|
fputs(p->progress_char, out);
|
|
for (; i < bar_width; i++)
|
|
fputc(' ', out);
|
|
|
|
fflush(out);
|
|
fputs("\e8\e[0K", out);
|
|
}
|
|
|
|
void apk_print_indented_init(struct apk_indent *i, struct apk_out *out, int err)
|
|
{
|
|
*i = (struct apk_indent) {
|
|
.f = err ? out->err : out->out,
|
|
.width = apk_out_get_width(out),
|
|
};
|
|
out->last_change++;
|
|
}
|
|
|
|
void apk_print_indented_line(struct apk_indent *i, const char *fmt, ...)
|
|
{
|
|
va_list va;
|
|
|
|
va_start(va, fmt);
|
|
vfprintf(i->f, fmt, va);
|
|
va_end(va);
|
|
i->x = i->indent = 0;
|
|
}
|
|
|
|
void apk_print_indented_group(struct apk_indent *i, int indent, const char *fmt, ...)
|
|
{
|
|
va_list va;
|
|
|
|
va_start(va, fmt);
|
|
i->x = vfprintf(i->f, fmt, va);
|
|
i->indent = indent ?: (i->x + 1);
|
|
if (fmt[strlen(fmt)-1] == '\n') i->x = 0;
|
|
va_end(va);
|
|
}
|
|
|
|
void apk_print_indented_end(struct apk_indent *i)
|
|
{
|
|
if (i->x) {
|
|
fprintf(i->f, "\n");
|
|
i->x = i->indent = 0;
|
|
}
|
|
}
|
|
|
|
int apk_print_indented(struct apk_indent *i, apk_blob_t blob)
|
|
{
|
|
if (i->x <= i->indent)
|
|
i->x += fprintf(i->f, "%*s" BLOB_FMT, i->indent - i->x, "", BLOB_PRINTF(blob));
|
|
else if (i->x + blob.len + 1 >= i->width)
|
|
i->x = fprintf(i->f, "\n%*s" BLOB_FMT, i->indent, "", BLOB_PRINTF(blob)) - 1;
|
|
else
|
|
i->x += fprintf(i->f, " " BLOB_FMT, BLOB_PRINTF(blob));
|
|
return 0;
|
|
}
|
|
|
|
void apk_print_indented_words(struct apk_indent *i, const char *text)
|
|
{
|
|
apk_blob_for_each_segment(APK_BLOB_STR(text), " ",
|
|
(apk_blob_cb) apk_print_indented, i);
|
|
}
|
|
|
|
void apk_print_indented_fmt(struct apk_indent *i, const char *fmt, ...)
|
|
{
|
|
char tmp[256];
|
|
size_t n;
|
|
va_list va;
|
|
|
|
va_start(va, fmt);
|
|
n = vsnprintf(tmp, sizeof(tmp), fmt, va);
|
|
apk_print_indented(i, APK_BLOB_PTR_LEN(tmp, n));
|
|
va_end(va);
|
|
}
|