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

dev.jbang.net.TrustedSources Maven / Gradle / Ivy

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

import java.io.File;
import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Set;
import java.util.regex.Pattern;
import java.util.stream.Collectors;

import com.google.gson.Gson;
import com.google.gson.JsonPrimitive;

import dev.jbang.Settings;
import dev.jbang.cli.ExitException;
import dev.jbang.source.ResourceRef;
import dev.jbang.util.TemplateEngine;
import dev.jbang.util.Util;

import io.quarkus.qute.Template;

public class TrustedSources {

	final static Pattern rLocalhost = Pattern.compile("(?i)^localhost(:\\d+)?$");
	final static Pattern r127 = Pattern.compile("(?i)^127.0.0.1(:\\d+)?$");

	private String[] trustedSources;
	private String[] temporaryTrustedSources;

	private static TrustedSources instance;

	TrustedSources(String[] trustedSources) {
		this.trustedSources = trustedSources;
		this.temporaryTrustedSources = new String[0];
	}

	public static TrustedSources load(Path toPath) throws IOException {
		String rawtd = Util.readString(toPath);
		Gson parser = new Gson();
		String[] strings = parser.fromJson(rawtd, String[].class);
		if (strings == null) {
			throw new IllegalStateException("Could not parse " + toPath);
		} else {
			return new TrustedSources(strings);
		}
	}

	private boolean isLocalhostAuthority(URI uri) {
		return uri.getScheme().equals("file") || rLocalhost.matcher(uri.getAuthority()).matches()
				|| r127.matcher(uri.getAuthority()).matches();
	}

	public String[] getTrustedSources() {
		return trustedSources;
	}

	/**
	 * Check whether a URL like https://www.microsoft.com matches the list of
	 * trusted sources.
	 *
	 * - Schemes must match - There's no subdomain matching. For example
	 * https://jbang.dev doesn't match https://www.jbang.dev - Star matches all
	 * subdomains. For example https://*.jbang.dev matches https://www.jbang.dev and
	 * https://foo.bar.jbangdev.com
	 */
	public boolean isURLTrusted(URI url) throws URISyntaxException {
		if (isLocalhostAuthority(url)) {
			return true;
		}

		boolean trusted = isURLTrusted(url, trustedSources) ||
				isURLTrusted(url, temporaryTrustedSources) ||
				// default trusted for usability and trust
				url.toString().startsWith("https://github.com/jbangdev/");

		return trusted;
	}

	private boolean isURLTrusted(URI url, String[] trustedSources) throws URISyntaxException {
		final String domain = String.format("%s://%s", url.getScheme(), url.getAuthority());

		for (String trustedSource : trustedSources) {
			if ("*".equals(trustedSource)) {
				return true;
			}

			if (domain.equals(trustedSource)) {
				return true;
			}

			URI parsedTrustedSource;
			if (trustedSource.startsWith("https://") || trustedSource.startsWith("http://")) {
				parsedTrustedSource = new URI(trustedSource);
				if (!url.getScheme().equals(parsedTrustedSource.getScheme())) {
					continue;
				}
			} else {
				parsedTrustedSource = new URI("https://" + trustedSource);
			}

			if (url.getAuthority().equals(parsedTrustedSource.getAuthority())) {
				if (pathMatches(url.getPath(), parsedTrustedSource.getPath())) {
					return true;
				} else {
					continue;
				}
			}

			if (trustedSource.contains("*")) {

				String[] reversedAuthoritySegments = reverse(url.getAuthority().split("\\."));

				String[] reversedTrustedSourceAuthoritySegments = reverse(
						parsedTrustedSource.getAuthority().split("\\."));

				boolean ruleIsSmaller = reversedTrustedSourceAuthoritySegments.length < reversedAuthoritySegments.length;
				boolean ruleHasStarAtEnd = reversedTrustedSourceAuthoritySegments[reversedTrustedSourceAuthoritySegments.length
						- 1].equals("*");
				if (ruleIsSmaller && ruleHasStarAtEnd) {
					reversedAuthoritySegments = Arrays.copyOfRange(reversedAuthoritySegments, 0,
							reversedTrustedSourceAuthoritySegments.length);
				}

				boolean authorityMatches = true;
				for (int i = 0; i < reversedAuthoritySegments.length; i++) {
					String val = reversedAuthoritySegments[i];
					String elementRule = reversedTrustedSourceAuthoritySegments[i];
					if (elementRule.equals("*")
							|| val.equals(elementRule)) {
						// they match
					} else {
						authorityMatches = false;
						break;
					}
				}

				if (authorityMatches && pathMatches(url.getPath(), parsedTrustedSource.getPath())) {
					return true;
				}
			}

		}

		// default trusted for usability and trust
		return url.toString().startsWith("https://github.com/jbangdev/");
	}

	boolean pathMatches(String open, String rule) {
		if ("/".equals(rule)) {
			return true;
		}

		if (rule.endsWith("/")) {
			rule = rule.substring(0, rule.length() - 1);
		}

		String[] openSegments = open.split("/");
		String[] ruleSegments = rule.split("/");
		for (int i = 0; i < ruleSegments.length; i++) {
			if (openSegments.length - 1 < i || !ruleSegments[i].equals(openSegments[i])) {
				return false;
			}
		}

		return true;
	}

	/**
	 * in place reverse of array
	 * 
	 * @param array array to reverse
	 * @param 
	 * @return reversed array
	 */
	static  T[] reverse(T[] array) {
		int i;
		T temp;
		for (i = 0; i < array.length / 2; i++) {
			temp = array[i];
			array[i] = array[array.length - i - 1];
			array[array.length - i - 1] = temp;
		}
		return array;
	}

	protected String getJSon(Collection rules) {

		rules = rules	.stream()
						.map(s -> new JsonPrimitive(s).toString())
						.collect(Collectors.toCollection(LinkedHashSet::new));

		String trustedsources = TemplateEngine	.instance()
												.getTemplate(
														ResourceRef.forResource("classpath:/trusted-sources.json.qute"))
												.data("trustedsources", rules)
												.render();
		return trustedsources;
	}

	public void addTemporary(String trust) {
		Util.infoMsg("Trusting for this run: " + trust);
		Set newrules = new LinkedHashSet<>(Arrays.asList(temporaryTrustedSources));
		if (newrules.add(trust)) {
			temporaryTrustedSources = newrules.toArray(new String[0]);
		} else {
			Util.warnMsg("Already trusted source(s). No changes made.");
		}
	}

	public void add(String trust) {
		add(trust, Settings.getTrustedSourcesFile().toFile());
	}

	public void add(List trust) {
		add(trust, Settings.getTrustedSourcesFile().toFile());
	}

	public void add(String trust, File storage) {
		add(Collections.singletonList(trust), storage);
	}

	public void add(List trust, File storage) {

		Util.infoMsg("Trusting permanently: " + trust);

		Set newrules = new LinkedHashSet<>(Arrays.asList(trustedSources));

		if (newrules.addAll(trust)) {
			trustedSources = newrules.toArray(new String[0]);
			save(newrules, storage);
		} else {
			Util.warnMsg("Already trusted source(s). No changes made.");
		}

	}

	public void remove(List trust, File storage) {

		Util.infoMsg("Removing permanent trust: " + trust);

		Set newrules = new LinkedHashSet<>(Arrays.asList(trustedSources));

		if (newrules.removeAll(trust)) {
			trustedSources = newrules.toArray(new String[0]);
			save(newrules, storage);
		} else {
			Util.warnMsg("Not found in trusted source(s). No changes made.");
		}

	}

	private void save(Set rules, File storage) {
		String newsources = getJSon(rules);
		try {
			Util.writeString(storage.toPath(), newsources);
		} catch (IOException e) {
			throw new ExitException(2, "Error when writing to " + storage, e);
		}
		trustedSources = rules.toArray(new String[0]);
	}

	public static void createTrustedSources() {
		Path trustedSourcesFile = Settings.getTrustedSourcesFile();
		if (Files.notExists(trustedSourcesFile)) {
			ResourceRef templateRef = ResourceRef.forResource("classpath:/trusted-sources.qute");
			Template template = TemplateEngine.instance().getTemplate(templateRef);
			if (template == null)
				throw new ExitException(1, "Could not locate template named: '" + templateRef + "'");
			String result = template.render();

			try {
				Util.writeString(trustedSourcesFile, result);
			} catch (IOException e) {
				Util.errorMsg("Could not create initial trusted-sources file at " + trustedSourcesFile, e);
			}

		}
	}

	public static TrustedSources instance() {
		if (instance == null) {
			Path trustedSourcesFile = Settings.getTrustedSourcesFile();
			if (Files.isRegularFile(trustedSourcesFile)) {
				try {
					instance = TrustedSources.load(trustedSourcesFile);
				} catch (IOException e) {
					Util.warnMsg("Could not read " + trustedSourcesFile);
					instance = new TrustedSources(new String[0]);
				}
			} else {
				instance = new TrustedSources(new String[0]);
			}
		}
		return instance;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy