All Downloads are FREE. Search and download functionalities are using the official Maven repository.

dev.jbang.cli.Catalog Maven / Gradle / Ivy

There is a newer version: 0.121.0
Show newest version
package dev.jbang.cli;

import java.io.PrintStream;
import java.io.PrintWriter;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;

import com.google.gson.Gson;
import com.google.gson.GsonBuilder;

import dev.jbang.Settings;
import dev.jbang.catalog.CatalogRef;
import dev.jbang.catalog.CatalogUtil;
import dev.jbang.source.ResourceRef;
import dev.jbang.util.ConsoleOutput;
import dev.jbang.util.Util;

import picocli.CommandLine;

@CommandLine.Command(name = "catalog", description = "Manage Catalogs of aliases.", subcommands = { CatalogAdd.class,
		CatalogUpdate.class, CatalogList.class, CatalogRemove.class })
public class Catalog {
}

abstract class BaseCatalogCommand extends BaseCommand {

	@CommandLine.Option(names = { "--global", "-g" }, description = "Use the global (user) catalog file")
	boolean global;

	@CommandLine.Option(names = { "--file", "-f" }, description = "Path to the catalog file to use")
	Path catalogFile;

	protected Path getCatalog(boolean strict) {
		Path cat;
		if (global) {
			cat = Settings.getUserCatalogFile();
		} else {
			if (catalogFile != null && Files.isDirectory(catalogFile)) {
				Path defaultCatalog = catalogFile.resolve(dev.jbang.catalog.Catalog.JBANG_CATALOG_JSON);
				Path hiddenCatalog = catalogFile.resolve(Settings.JBANG_DOT_DIR)
												.resolve(dev.jbang.catalog.Catalog.JBANG_CATALOG_JSON);
				if (!Files.exists(defaultCatalog) && Files.exists(hiddenCatalog)) {
					cat = hiddenCatalog;
				} else {
					cat = defaultCatalog;
				}
			} else {
				cat = catalogFile;
			}
			if (strict && cat != null && !Files.isRegularFile(cat)) {
				throw new IllegalArgumentException("Catalog file not found at: " + catalogFile);
			}
		}
		return cat;
	}
}

@CommandLine.Command(name = "add", description = "Add a catalog.")
class CatalogAdd extends BaseCatalogCommand {

	@CommandLine.Option(names = { "--description",
			"-d" }, description = "A description for the catalog")
	String description;

	@CommandLine.Option(names = { "--name" }, description = "A name for the catalog")
	String name;

	@CommandLine.Option(names = {
			"--force" }, description = "Force overwriting of existing catalog")
	boolean force;

	@CommandLine.Option(names = {
			"--import" }, description = "Import catalog items into the catalog's scope")
	Boolean importItems;

	@CommandLine.Parameters(paramLabel = "urlOrFile", index = "0", description = "A file or URL to a catalog file", arity = "1")
	String urlOrFile;

	@Override
	public Integer doCall() {
		if (name != null && !dev.jbang.catalog.Catalog.isValidName(name)) {
			throw new IllegalArgumentException(
					"Invalid catalog name, it should start with a letter followed by 0 or more letters, digits, underscores, hyphens or dots");
		}
		if (name == null) {
			name = CatalogUtil.nameFromRef(urlOrFile);
		}
		CatalogRef ref = CatalogRef.get(urlOrFile);
		Path catFile = getCatalog(false);
		if (catFile == null) {
			catFile = dev.jbang.catalog.Catalog.getCatalogFile(null);
		}
		if (force || !CatalogUtil.hasCatalogRef(catFile, name)) {
			CatalogUtil.addCatalogRef(catFile, name, ref.catalogRef, ref.description, importItems);
		} else {
			Util.infoMsg("A catalog with name '" + name + "' already exists, use '--force' to add anyway.");
			return EXIT_INVALID_INPUT;
		}
		info(String.format("Catalog '%s' added to '%s'", name, catFile));
		return EXIT_OK;
	}
}

@CommandLine.Command(name = "update", description = "Retrieve the latest contents of the catalogs.")
class CatalogUpdate extends BaseCatalogCommand {

	@Override
	public Integer doCall() {
		PrintWriter err = spec.commandLine().getErr();
		Map cats = dev.jbang.catalog.Catalog.getMerged(false, true).catalogs;
		cats
			.entrySet()
			.stream()
			.forEach(e -> {
				String ref = e.getValue().catalogRef;
				err.println("Updating catalog '" + e.getKey() + "' from " + ref + "...");
				Util.freshly(() -> {
					try {
						dev.jbang.catalog.Catalog.getByRef(ref);
					} catch (Exception ex) {
						Util.warnMsg("Unable to read catalog " + ref + " (referenced from "
								+ e.getValue().catalog.catalogRef + ")");
					}
					return null;
				});
			});
		return EXIT_OK;
	}
}

@CommandLine.Command(name = "list", description = "Show currently defined catalogs.")
class CatalogList extends BaseCatalogCommand {

	@CommandLine.Option(names = { "--show-origin" }, description = "Show the origin of the catalog")
	boolean showOrigin;

	@CommandLine.Parameters(paramLabel = "name", index = "0", description = "The name of a catalog", arity = "0..1")
	String name;

	@CommandLine.Mixin
	FormatMixin formatMixin;

	private static final int INDENT_SIZE = 3;

	@Override
	public Integer doCall() {
		PrintStream out = System.out;
		if (name == null) {
			dev.jbang.catalog.Catalog catalog;
			Path cat = getCatalog(true);
			if (cat != null) {
				catalog = dev.jbang.catalog.Catalog.get(cat);
			} else {
				catalog = dev.jbang.catalog.Catalog.getMerged(true, false);
			}
			if (showOrigin) {
				printCatalogsWithOrigin(out, name, catalog, formatMixin.format);
			} else {
				printCatalogs(out, name, catalog, formatMixin.format);
			}
		} else {
			dev.jbang.catalog.Catalog catalog = dev.jbang.catalog.Catalog.getByName(name);
			if (formatMixin.format == FormatMixin.Format.json) {
				List aliasCats = AliasList.getAliasesWithOrigin(name, catalog);
				List tplCats = TemplateList.getTemplatesWithOrigin(name, catalog, false, false);
				List catCats = getCatalogsWithOrigin(name, catalog);
				Map cats = new HashMap<>();
				for (CatalogOut cat : aliasCats) {
					cats.put(cat.resourceRef, cat);
				}
				for (CatalogOut cat : tplCats) {
					if (cats.containsKey(cat.resourceRef)) {
						cats.get(cat.resourceRef).templates = cat.templates;
					} else {
						cats.put(cat.resourceRef, cat);
					}
				}
				for (CatalogOut cat : catCats) {
					if (cats.containsKey(cat.resourceRef)) {
						cats.get(cat.resourceRef).catalogs = cat.catalogs;
					} else {
						cats.put(cat.resourceRef, cat);
					}
				}
				Gson parser = new GsonBuilder().disableHtmlEscaping().setPrettyPrinting().create();
				parser.toJson(cats.values(), out);
			} else {
				if (!catalog.aliases.isEmpty()) {
					out.println("Aliases:");
					out.println("--------");
					AliasList.printAliases(out, name, catalog, FormatMixin.Format.text);
				}
				if (!catalog.templates.isEmpty()) {
					out.println("Templates:");
					out.println("----------");
					TemplateList.printTemplates(out, name, catalog, false, false, FormatMixin.Format.text);
				}
				if (!catalog.catalogs.isEmpty()) {
					out.println("Catalogs:");
					out.println("---------");
					printCatalogs(out, name, catalog, FormatMixin.Format.text);
				}
			}
		}
		return EXIT_OK;
	}

	static void printCatalogs(PrintStream out, String catalogName, dev.jbang.catalog.Catalog catalog,
			FormatMixin.Format format) {
		List catalogs = catalog.catalogs
														.keySet()
														.stream()
														.sorted()
														.map(name -> getCatalogRefOut(catalogName, catalog, name))
														.collect(Collectors.toList());

		if (format == FormatMixin.Format.json) {
			Gson parser = new GsonBuilder().disableHtmlEscaping().setPrettyPrinting().create();
			parser.toJson(catalogs, out);
		} else {
			catalogs.forEach(c -> printCatalogRef(out, c, 0));
		}
	}

	static List getCatalogsWithOrigin(String catalogName, dev.jbang.catalog.Catalog catalog) {
		Map> groups = catalog.catalogs
																		.keySet()
																		.stream()
																		.sorted()
																		.map(name -> getCatalogRefOut(catalogName,
																				catalog, name))
																		.collect(Collectors.groupingBy(
																				c -> c._catalogRef));
		return groups	.entrySet()
						.stream()
						.map(e -> new CatalogOut(null, e.getKey(),
								null, null, e.getValue()))
						.collect(Collectors.toList());
	}

	static void printCatalogsWithOrigin(PrintStream out, String catalogName, dev.jbang.catalog.Catalog catalog,
			FormatMixin.Format format) {
		List catalogs = getCatalogsWithOrigin(catalogName, catalog);
		if (format == FormatMixin.Format.json) {
			Gson parser = new GsonBuilder().disableHtmlEscaping().setPrettyPrinting().create();
			parser.toJson(catalogs, out);
		} else {
			catalogs.forEach(cat -> {
				out.println(ConsoleOutput.bold(dev.jbang.catalog.Catalog.simplifyRef(cat.resourceRef)));
				cat.catalogs.forEach(c -> printCatalogRef(out, c, 1));
			});
		}
	}

	static class CatalogOut {
		public String name;
		public String resourceRef;
		public String description;
		public List aliases;
		public List templates;
		public List catalogs;

		public CatalogOut(String name, ResourceRef ref, List aliases,
				List templates, List catalogs) {
			this.name = name;
			if (name == null) {
				if (aliases != null) {
					this.name = aliases.get(0).catalogName;
				}
				if (templates != null) {
					this.name = templates.get(0).catalogName;
				}
				if (catalogs != null) {
					this.name = catalogs.get(0).catalogName;
				}
			}
			resourceRef = ref.getOriginalResource();
			this.aliases = aliases;
			this.templates = templates;
			this.catalogs = catalogs;
		}
	}

	static class CatalogRefOut {
		public String name;
		public String catalogName;
		public String fullName;
		public String catalogRef;
		public String description;
		public boolean importItems;

		public transient ResourceRef _catalogRef;
	}

	private static CatalogRefOut getCatalogRefOut(String catalogName, dev.jbang.catalog.Catalog catalog, String name) {
		CatalogRef ref = catalog.catalogs.get(name);
		String catName = catalogName != null ? dev.jbang.catalog.Catalog.simplifyRef(catalogName)
				: CatalogUtil.catalogRef(name);
		String fullName = catalogName != null ? name + "@" + catName : name;

		CatalogRefOut out = new CatalogRefOut();
		out.name = name;
		out.catalogName = catName;
		out.fullName = fullName;
		out.catalogRef = ref.catalogRef;
		out.description = ref.description;
		out.importItems = Boolean.TRUE.equals(ref.importItems);
		out._catalogRef = ref.catalog.catalogRef;
		return out;
	}

	private static void printCatalogRef(PrintStream out, CatalogRefOut catalogRef, int indent) {
		String prefix1 = Util.repeat(" ", indent * INDENT_SIZE);
		String prefix2 = Util.repeat(" ", (indent + 1) * INDENT_SIZE);
		out.println(prefix1 + getColoredFullName(catalogRef.fullName)
				+ (catalogRef.importItems ? ConsoleOutput.magenta(" [importing]") : ""));
		if (catalogRef.description != null) {
			out.println(prefix2 + catalogRef.description);
		}
		out.println(prefix2 + catalogRef.catalogRef);
	}

	static String getColoredFullName(String fullName) {
		StringBuilder res = new StringBuilder();
		String[] parts = fullName.split("@");
		res.append(ConsoleOutput.yellow(parts[0]));
		for (int i = 1; i < parts.length; i++) {
			res.append(ConsoleOutput.cyan("@"));
			res.append(parts[i]);
		}
		return res.toString();
	}
}

@CommandLine.Command(name = "remove", description = "Remove existing catalog.")
class CatalogRemove extends BaseCatalogCommand {

	@CommandLine.Parameters(paramLabel = "name", index = "0", description = "The name of the catalog", arity = "1")
	String name;

	@Override
	public Integer doCall() {
		Path cat = getCatalog(true);
		if (cat != null) {
			CatalogUtil.removeCatalogRef(cat, name);
		} else {
			CatalogUtil.removeNearestCatalogRef(name);
		}
		return EXIT_OK;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy