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

org.appng.api.PathInfo Maven / Gradle / Ivy

/*
 * Copyright 2011-2021 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.appng.api;

import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

import javax.servlet.http.HttpServletRequest;

import org.apache.commons.lang3.StringUtils;
import org.appng.api.model.Application;
import org.appng.api.model.Site;

/**
 * Default {@link Path}-implementation
 * 
 * @author Matthias Müller
 */
public class PathInfo implements Path {

	private static final String OUTPUT_PREFIX = "_";
	private static final String DOT = ".";
	private final String guiPath;
	private final String servicePath;
	private final List blobDirectories;
	private final List documentDirectories;
	private String repositoryPath;
	private String monitoringPath;
	private final String host;
	private String rootPath;
	private String servletPath;
	private String applicationName;
	private String actionName;
	private String actionValue;
	private String currentSite;
	private List pathElements;
	private List jspUrlParams = null;

	private int siteIdx = -1;
	private int applicationIdx = -1;
	private int pageIdx = -1;
	private String page;
	private int rootIdx;
	private String extension;
	private String domain;

	/**
	 * Creates a new {@link PathInfo}
	 * 
	 * @param host
	 *                            the host of the current {@link org.appng.api.model.Site}
	 * @param domain
	 *                            the domain of the current {@link org.appng.api.model.Site}
	 * @param currentSite
	 *                            the name of the current {@link org.appng.api.model.Site}
	 * @param servletPath
	 *                            the current servletPath, as returned by
	 *                            {@link javax.servlet.http.HttpServletRequest#getServletPath()}
	 * @param guiPath
	 *                            value of the property {@value org.appng.api.SiteProperties#MANAGER_PATH} of the
	 *                            current {@link org.appng.api.model.Site}
	 * @param servicePath
	 *                            value of the property {@value org.appng.api.SiteProperties#SERVICE_PATH} of the
	 *                            current {@link org.appng.api.model.Site}
	 * @param blobDirectories
	 *                            a list parsed from the property {@value org.appng.api.SiteProperties#ASSETS_DIR} of
	 *                            the current {@link org.appng.api.model.Site}
	 * @param documentDirectories
	 *                            a list parsed from the property {@value org.appng.api.SiteProperties#DOCUMENT_DIR} of
	 *                            the current {@link org.appng.api.model.Site}
	 * @param repositoryPath
	 *                            value of the platform property
	 *                            {@value org.appng.api.Platform.Property#REPOSITORY_PATH}
	 * @param monitoringPath
	 *                            the path to appNGs built in monitoring
	 * @param extension
	 *                            value of the platform property {@value org.appng.api.Platform.Property#JSP_FILE_TYPE}
	 */
	public PathInfo(String host, String domain, String currentSite, String servletPath, String guiPath,
			String servicePath, List blobDirectories, List documentDirectories, String repositoryPath,
			String monitoringPath, String extension) {
		this.host = host;
		this.domain = domain;
		this.currentSite = currentSite;
		this.servletPath = servletPath;
		this.guiPath = guiPath;
		this.servicePath = servicePath;
		this.blobDirectories = blobDirectories;
		this.documentDirectories = documentDirectories;
		this.repositoryPath = repositoryPath;
		this.monitoringPath = monitoringPath;
		this.extension = extension;
		parse();
	}

	/**
	 * @see PathInfo#PathInfo(String, String, String, String, String, String, List, List, String, String, String)
	 */
	public PathInfo(String host, String domain, String currentSite, String servletPath, String guiPath,
			String servicePath, List blobDirectories, List documentDirectories, String repositoryPath,
			String extension) {
		this(host, domain, currentSite, servletPath, guiPath, servicePath, blobDirectories, documentDirectories,
				repositoryPath, "/health", extension);
	}

	private void parse() {
		if (null == servletPath) {
			this.servletPath = "";
		}
		this.pathElements = splitPath(servletPath);

		this.rootIdx = 1;
		if (hasElementAt(rootIdx)) {
			this.rootPath = SEPARATOR + pathElements.get(rootIdx);
		} else {
			this.rootPath = servletPath;
		}
		if (isGui()) {
			int idx = pathElements.indexOf(guiPath.substring(1));
			if (hasElementAt(idx) && hasElementAt(idx + 1)) {
				siteIdx = idx + 1;
				if (pathElements.get(siteIdx).startsWith(OUTPUT_PREFIX)) {
					siteIdx += 1;
					if (pathElements.get(siteIdx).startsWith(OUTPUT_PREFIX)) {
						siteIdx += 1;
					}
				}

				if (hasElementAt(siteIdx + 1)) {
					applicationIdx = siteIdx + 1;
					if (hasElementAt(applicationIdx + 1)) {
						pageIdx = applicationIdx + 1;
					}
				}
			}
		} else if (isService()) {
			int idx = pathElements.indexOf(servicePath.substring(1));
			if (hasElementAt(idx + 1)) {
				this.siteIdx = idx + 1;
			}
			if (hasElementAt(idx + 2)) {
				this.applicationIdx = idx + 2;
			}
		}
	}

	private List splitPath(String path) {
		List elements = new ArrayList<>(Arrays.asList(path.split(SEPARATOR)));
		if (elements.size() > 0) {
			int lastIdx = elements.size() - 1;
			String lastElement = elements.get(lastIdx);
			int anchorIndex = lastElement.indexOf('#');
			if (anchorIndex > 0) {
				elements.remove(lastIdx);
				elements.add(lastElement.substring(0, anchorIndex));
			}
		}
		return elements;
	}

	public boolean hasElementAt(int idx) {
		return idx > -1 && idx < pathElements.size();
	}

	public String getLastElement() {
		return getElementAt(getElementCount() - 1);
	}

	public String getElementAt(int idx) {
		if (hasElementAt(idx)) {
			return pathElements.get(idx);
		}
		return null;
	}

	public void checkPathLength(int minLength) throws IOException {
		if (pathElements.size() < minLength) {
			throw new IOException("invalid path");
		}
	}

	public boolean isStaticContent() {
		return blobDirectories.contains(rootPath);
	}

	public boolean isDocument() {
		return documentDirectories.contains(rootPath);
	}

	public boolean isGui() {
		return hasElementAt(1) && pathElements.get(1).equals(guiPath.substring(1));
	}

	public boolean isService() {
		return hasElementAt(1) && pathElements.get(1).equals(servicePath.substring(1));
	}

	public boolean isJsp() {
		return servletPath != null && servletPath.endsWith(DOT + extension);
	}

	public String getSiteName() {
		if (hasSite()) {
			return pathElements.get(siteIdx);
		} else {
			return currentSite;
		}
	}

	public boolean hasSite() {
		return hasElementAt(siteIdx);
	}

	/**
	 * Manually sets the name of the selected {@link Application}
	 * 
	 * @param application
	 *                    the name of the {@link Application}
	 */
	public void setApplicationName(String application) {
		this.applicationName = application;
		this.applicationIdx = pathElements.indexOf(application);
		this.pageIdx = -1;
	}

	public String getApplicationName() {
		if (null != applicationName) {
			return applicationName;
		} else if (hasApplication()) {
			return pathElements.get(applicationIdx);
		}
		return null;
	}

	public boolean hasApplication() {
		return hasElementAt(applicationIdx);
	}

	/**
	 * Manually sets the page within a {@link Application}
	 * 
	 * @param page
	 *             the page to set
	 */
	public void setPage(String page) {
		this.page = page;
		this.pageIdx = pathElements.indexOf(page);
	}

	public String getPage() {
		if (null == this.page && hasElementAt(pageIdx)) {
			return pathElements.get(pageIdx);
		}
		return page;
	}

	/**
	 * Manually sets the name and value for an action within a {@link Application}
	 * 
	 * @param actionName
	 *                    the name of the action
	 * @param actionValue
	 *                    the value for the action
	 */
	public void setAction(String actionName, String actionValue) {
		this.actionName = actionName;
		this.actionValue = actionValue;
	}

	public String getActionName() {
		return actionName;
	}

	public String getActionValue() {
		return actionValue;
	}

	public boolean hasAction() {
		return StringUtils.isNotEmpty(actionName) && StringUtils.isNotEmpty(actionValue);
	}

	private String initJspUrlParameters(File wwwRootFile) {
		String realServletPath = servletPath;
		PathSegmenter segmenter = new PathSegmenter(servletPath);
		for (int i = 1; i <= segmenter.size(); i++) {
			realServletPath = segmenter.getSegments(i);
			File jspFile = new File(wwwRootFile, realServletPath + DOT + extension);
			if (jspFile.exists()) {
				jspUrlParams = segmenter.getSegmentsList(i, segmenter.size());
				realServletPath += DOT + extension;
				break;
			}
		}
		return realServletPath;
	}

	public List getApplicationUrlParameters() {
		if (isGui()) {
			if (hasElementAt(pageIdx) && hasElementAt(pageIdx + 1)) {
				return pathElements.subList(pageIdx + 1, pathElements.size());
			}
		} else if (isService() && hasElementAt(applicationIdx) && hasElementAt(applicationIdx + 3)) {
			return pathElements.subList(applicationIdx + 3, pathElements.size());
		}
		return new ArrayList<>(0);
	}

	public List getJspUrlParameters() {
		if (null == jspUrlParams) {
			jspUrlParams = new ArrayList<>();
		}
		return jspUrlParams;
	}

	public String getRootPath() {
		return rootPath;
	}

	public String getHost() {
		return host;
	}

	public String getDomain() {
		return domain;
	}

	public String getServletPath() {
		return servletPath;
	}

	public String getCurrentPath() {
		if (isGui() && isRoot()) {
			String result = rootPath + SEPARATOR + getSiteName();
			if (getApplicationName() != null) {
				result += SEPARATOR + getApplicationName();
			}
			return result;
		}
		return getServletPath();
	}

	public boolean isRoot() {
		return servletPath.equals(rootPath);
	}

	public boolean isRootIgnoreTrailingSlash() {
		if (isRoot() || (servletPath.equals(rootPath + SEPARATOR))) {
			return true;
		} else {
			return false;
		}
	}

	public boolean isRepository() {
		return servletPath.startsWith(SEPARATOR + repositoryPath);
	}

	@Override
	public boolean isMonitoring() {
		return servletPath.startsWith(monitoringPath);
	}

	public String getGuiPath() {
		return guiPath;
	}

	public List getBlobDirectories() {
		return blobDirectories;
	}

	public List getDocumentDirectories() {
		return documentDirectories;
	}

	public String getPlatformUrl() {
		return domain + servletPath;
	}

	public String getOutputFormat() {
		if (isGui() && siteIdx - rootIdx > 1) {
			return pathElements.get(rootIdx + 1).substring(1);
		}
		return null;
	}

	public String getOutputType() {
		if (isGui() && siteIdx - rootIdx > 2) {
			return pathElements.get(rootIdx + 2).substring(1);
		}
		return null;
	}

	public boolean hasOutputFormat() {
		return null != getOutputFormat();
	}

	public boolean hasOutputType() {
		return null != getOutputType();
	}

	public String getService() {
		if (isService() && hasElementAt(applicationIdx) && hasElementAt(applicationIdx + 2)) {
			String parameters = pathElements.get(applicationIdx + 2);
			int indexOf = parameters.indexOf('?');
			if (indexOf > 0) {
				return parameters.substring(0, indexOf);
			}
			return parameters;
		}
		return null;
	}

	public String getServicePath() {
		return servicePath;
	}

	public boolean isPathSelected(String path) {
		List pathChunks = splitPath(path);
		if (getElementCount() >= pathChunks.size()) {
			for (int i = 0; i < pathChunks.size(); i++) {
				if (!pathChunks.get(i).equals(getElementAt(i))) {
					return false;
				}
			}
			return true;
		}
		return false;
	}

	public String getOutputPrefix() {
		String prefix = "";
		if (hasOutputFormat()) {
			prefix += SEPARATOR + OUTPUT_PREFIX + getOutputFormat();
			if (hasOutputType()) {
				prefix += SEPARATOR + OUTPUT_PREFIX + getOutputType();
			}
		}
		return prefix;
	}

	public String getExtension() {
		return extension;
	}

	/**
	 * Builds the the path to which a {@link HttpServletRequest} has to be forwarded to in order the retrieve a file
	 * from a document directory. If the requested fiel is a JSP, the JSP Url-Parameters are being initialized.
	 * 
	 * @param wwwRootPath
	 *                    the relative path to a {@link Site}s web-folder, under which the document-folders reside
	 * @param wwwRootFile
	 *                    a file representing the very same relative path
	 * 
	 * @return the forward path
	 * 
	 * @see #isDocument()
	 * @see #getDocumentDirectories()
	 * @see #isJsp()
	 */
	public String getForwardPath(String wwwRootPath, File wwwRootFile) {
		String realServletPath = initJspUrlParameters(wwwRootFile);
		return wwwRootPath + realServletPath;
	}

	public int getApplicationIndex() {
		return applicationIdx;
	}

	public int getElementCount() {
		return pathElements.size();
	}

	class PathSegmenter {

		private final List segments;

		public PathSegmenter(String servletPath) {
			this.segments = Arrays.asList(servletPath.split(SEPARATOR));
		}

		public int size() {
			return segments.size();
		}

		public List getSegmentsList(int fromIndex, int toIndex) {
			return segments.subList(fromIndex, toIndex);
		}

		public String getSegments(int index) {
			StringBuilder segmentString = new StringBuilder();
			getSegmentsList(1, index).forEach(segment -> segmentString.append(SEPARATOR + segment));
			return 0 == segmentString.length() ? SEPARATOR : segmentString.toString();
		}
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy