/* print.c - Alpine Package Keeper (APK) * * Copyright (C) 2005-2008 Natanael Copa * Copyright (C) 2008-2011 Timo Teräs * All rights reserved. * * SPDX-License-Identifier: GPL-2.0-only */ #include #include #include #include #include #include #include #include #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); }