/* dot.c - Alpine Package Keeper (APK)
*
* Copyright (C) 2008-2011 Timo Teräs <timo.teras@iki.fi>
* All rights reserved.
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation. See http://www.gnu.org/ for details.
*/
#include <stdio.h>
#include <fnmatch.h>
#include "apk_applet.h"
#include "apk_database.h"
#include "apk_print.h"
#define S_EVALUATED -1
#define S_EVALUATING -2
struct dot_ctx {
int errors_only;
int not_empty;
};
static int dot_parse(void *pctx, struct apk_db_options *dbopts,
int optch, int optindex, const char *optarg)
{
struct dot_ctx *ctx = (struct dot_ctx *) pctx;
switch (optch) {
case 0x10000:
ctx->errors_only = 1;
break;
default:
return -1;
}
return 0;
}
static void start_graph(struct dot_ctx *ctx)
{
if (ctx->not_empty)
return;
ctx->not_empty = 1;
printf( "digraph \"apkindex\" {\n"
" rankdir=LR;\n"
" node [shape=box];\n");
}
static int dump_pkg(struct dot_ctx *ctx, struct apk_package *pkg)
{
int i, j, r, ret = 0;
if (pkg->state_int == S_EVALUATED)
return 0;
if (pkg->state_int <= S_EVALUATING) {
pkg->state_int--;
return 1;
}
pkg->state_int = S_EVALUATING;
for (i = 0; i < pkg->depends->num; i++) {
struct apk_dependency *dep = &pkg->depends->item[i];
struct apk_name *name = dep->name;
if (name->providers->num == 0) {
start_graph(ctx);
printf(" \"" PKG_VER_FMT "\" -> \"%s\" [color=red];\n",
PKG_VER_PRINTF(pkg),
name->name);
if (!name->state_int) {
printf(" \"%s\" [style=dashed, color=red, fontcolor=red, shape=octagon];\n",
name->name);
name->state_int = 1;
}
} else {
for (j = 0; j < name->providers->num; j++) {
struct apk_provider *p0 = &name->providers->item[j];
if (!apk_dep_is_provided(dep, p0))
continue;
r = dump_pkg(ctx, p0->pkg);
ret += r;
if (r || (!ctx->errors_only)) {
start_graph(ctx);
if (p0->pkg->name != dep->name) {
/* provided package */
printf(" \"" PROVIDER_FMT "\" -> \"" PKG_VER_FMT "\"[arrowhead=inv,color=green];\n",
PROVIDER_PRINTF(p0),
PKG_VER_PRINTF(p0->pkg));
}
printf(" \"" PKG_VER_FMT "\" -> \"" PROVIDER_FMT "\"%s;\n",
PKG_VER_PRINTF(pkg),
PROVIDER_PRINTF(p0),
r ? "[color=red]" : "");
}
}
}
}
ret -= S_EVALUATING - pkg->state_int;
pkg->state_int = S_EVALUATED;
return ret;
}
static int foreach_pkg(apk_hash_item item, void *ctx)
{
dump_pkg((struct dot_ctx *) ctx, (struct apk_package *) item);
return 0;
}
static int dot_main(void *pctx, struct apk_database *db, int argc, char **argv)
{
struct dot_ctx *ctx = (struct dot_ctx *) pctx;
int i, j;
if (argc) {
for (i = 0; i < argc; i++) {
struct apk_name *name = apk_db_get_name(db, APK_BLOB_STR(argv[i]));
if (!name)
continue;
for (j = 0; j < name->providers->num; j++)
dump_pkg(ctx, name->providers->item[j].pkg);
}
} else {
apk_hash_foreach(&db->available.packages, foreach_pkg, pctx);
}
if (!ctx->not_empty)
return 1;
printf("}\n");
return 0;
}
static struct apk_option dot_options[] = {
{ 0x10000, "errors", "Output only parts of the graph which are considered "
"errorneus: e.g. cycles and missing packages" },
};
static struct apk_applet apk_dot = {
.name = "dot",
.help = "Generate graphviz graphs",
.arguments = "PKGMASK...",
.open_flags = APK_OPENF_READ | APK_OPENF_NO_STATE,
.context_size = sizeof(struct dot_ctx),
.num_options = ARRAY_SIZE(dot_options),
.options = dot_options,
.parse = dot_parse,
.main = dot_main,
};
APK_DEFINE_APPLET(apk_dot);