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

fiftyone.pipeline.javascriptbuilder.flowelements.JavaScriptBuilderElement Maven / Gradle / Ivy

/* *********************************************************************
 * This Original Work is copyright of 51 Degrees Mobile Experts Limited.
 * Copyright 2023 51 Degrees Mobile Experts Limited, Davidson House,
 * Forbury Square, Reading, Berkshire, United Kingdom RG1 3EU.
 *
 * This Original Work is licensed under the European Union Public Licence
 * (EUPL) v.1.2 and is subject to its terms as set out below.
 *
 * If a copy of the EUPL was not distributed with this file, You can obtain
 * one at https://opensource.org/licenses/EUPL-1.2.
 *
 * The 'Compatible Licences' set out in the Appendix to the EUPL (as may be
 * amended by the European Commission) shall be deemed incompatible for
 * the purposes of the Work and the provisions of the compatibility
 * clause in Article 5 of the EUPL shall not apply.
 *
 * If using the Work as, or as part of, a network application, by
 * including the attribution notice(s) required under Article 5 of the EUPL
 * in the end user terms of the application under an appropriate heading,
 * such notice(s) shall fulfill the requirements of that article.
 * ********************************************************************* */

package fiftyone.pipeline.javascriptbuilder.flowelements;

import com.github.mustachejava.DefaultMustacheFactory;
import com.github.mustachejava.Mustache;
import com.github.mustachejava.MustacheFactory;
import fiftyone.pipeline.core.data.*;
import fiftyone.pipeline.core.data.factories.ElementDataFactory;
import fiftyone.pipeline.core.exceptions.PipelineConfigurationException;
import fiftyone.pipeline.core.exceptions.PipelineDataException;
import fiftyone.pipeline.core.flowelements.FlowElementBase;
import fiftyone.pipeline.engines.data.AspectPropertyValue;
import fiftyone.pipeline.engines.exceptions.PropertyMissingException;
import fiftyone.pipeline.javascriptbuilder.Constants;
import fiftyone.pipeline.javascriptbuilder.data.JavaScriptBuilderData;
import fiftyone.pipeline.javascriptbuilder.templates.JavaScriptResource;
import fiftyone.pipeline.jsonbuilder.data.JsonBuilderData;
import org.json.JSONObject;
import org.slf4j.Logger;

import java.io.*;
import java.net.URLEncoder;
import java.util.*;

import static fiftyone.pipeline.core.Constants.*;
import static fiftyone.pipeline.engines.fiftyone.flowelements.Constants.EVIDENCE_SEQUENCE;
import static fiftyone.pipeline.engines.fiftyone.flowelements.Constants.EVIDENCE_SESSIONID;
import static fiftyone.pipeline.javascriptbuilder.Constants.EVIDENCE_ENABLE_COOKIES;
import static fiftyone.pipeline.javascriptbuilder.Constants.EVIDENCE_OBJECT_NAME;

//! [class]

/**
 * JavaScript Builder Element generates a JavaScript include to be run on the
 * client device.
 * @see Specification
 */
public class JavaScriptBuilderElement
    extends FlowElementBase {

    protected String host;
    protected String endpoint;
    protected String protocol;
    protected String contextRoot;
    protected final String objName;
    protected final boolean enableCookies;
    private final Mustache mustache;
    
    //! [constructor]
    /**
     * Default constructor.
     * @param logger The logger.
     * @param elementDataFactory The element data factory.
     * @param endpoint Set the endpoint which will be queried on the host.
     *                 e.g /api/v4/json
     * @param objName The default name of the object instantiated by the client
     *                JavaScript.
     * @param enableCookies Set whether the client JavaScript stored results of
     * client side processing in cookies.
     * @param host The host that the client JavaScript should query for updates.
     * If null or blank then the host from the request will be used
     * @param protocol The protocol (HTTP or HTTPS) that the client JavaScript
     *                 will use when querying for updates. If null or blank
     *                 then the protocol from the request will be used
     */
    public JavaScriptBuilderElement(
            Logger logger,
            ElementDataFactory elementDataFactory,
            String endpoint,
            String objName,
            boolean enableCookies,
            String host,
            String protocol) {
        this(logger, elementDataFactory, endpoint, objName, enableCookies, host, protocol, null);        
    }
    //! [constructor]
    
    //! [constructor]
    /**
     * Default constructor.
     * @param logger The logger.
     * @param elementDataFactory The element data factory.
     * @param endpoint Set the endpoint which will be queried on the host.
     *                 e.g /api/v4/json
     * @param objName The default name of the object instantiated by the client
     *                JavaScript.
     * @param enableCookies Set whether the client JavaScript stored results of
     * client side processing in cookies.
     * @param host The host that the client JavaScript should query for updates.
     * If null or blank then the host from the request will be used
     * @param protocol The protocol (HTTP or HTTPS) that the client JavaScript
     *                 will use when querying for updates. If null or blank
     *                 then the protocol from the request will be used
     * @param contextRoot The <context-root> setting from the web.xml.
     *                 This is needed when creating the callback URL.
     */
    public JavaScriptBuilderElement(
            Logger logger,
            ElementDataFactory elementDataFactory,
            String endpoint,
            String objName,
            boolean enableCookies,
            String host,
            String protocol,
            String contextRoot) {
        super(logger, elementDataFactory);
        
        MustacheFactory mf = new DefaultMustacheFactory();
        InputStream in = getClass().getResourceAsStream(Constants.TEMPLATE);
        BufferedReader reader = new BufferedReader(new InputStreamReader(in));
        mustache = mf.compile(reader, "template");
        
        this.host = host;
        this.endpoint = endpoint;
        this.protocol = protocol;
        this.objName = objName.isEmpty() ? Constants.DEFAULT_OBJECT_NAME : objName;
        this.enableCookies = enableCookies;
        this.contextRoot = contextRoot;
    }
    //! [constructor]

    @Override
    protected void processInternal(FlowData data) throws Exception {
        String reqHost = getHost(data);
        String reqProtocol = getProtocol(data);
        boolean supportsPromises = getSupportsPromises(data);

        // Try and get the web server context root evidence so it can be 
        // used to construct the correct path for the Json refresh.
        if (contextRoot == null || contextRoot.isEmpty()) {
            contextRoot = getContextRoot(data);
        }

        // Get the JSON include to embed into the JavaScript include.
        String jsonObject = getJsonObject(data);
        // Generate any required parameters for the JSON request.
        Map parameters = getParameters(data);
        String queryParams = getQueryParams(parameters);

        String url = getUrl(reqProtocol, reqHost, queryParams);

        String sessionId = getSessionId(data);
        Integer sequence = getSequence(data);
        String serializedParameters = serializeParameters(parameters);

        // With the gathered resources, build a new JavaScriptResource.
        buildJavaScript(data, jsonObject, supportsPromises, url, serializedParameters, sessionId, sequence);
    }

    @Override
    public String getElementDataKey() {
        return "javascript-builder";
    }

    @Override
    public EvidenceKeyFilter getEvidenceKeyFilter() {
        return new EvidenceKeyFilterWhitelist(Arrays.asList(
                Constants.EVIDENCE_HOST_KEY,
                fiftyone.pipeline.core.Constants.EVIDENCE_PROTOCOL,
                EVIDENCE_OBJECT_NAME,
                EVIDENCE_ENABLE_COOKIES,
                EVIDENCE_WEB_CONTEXT_ROOT),
                String.CASE_INSENSITIVE_ORDER);
    }

    @Override
    public List getProperties() {
        return Collections.singletonList(
                new ElementPropertyMetaDataDefault(
                        "javascript",
                        this,
                        "javascript",
                        String.class,
                        true));
    }

    @Override
    protected void managedResourcesCleanup() {
        // Nothing to clean up here.
    }

    @Override
    protected void unmanagedResourcesCleanup() {
        // Nothing to clean up here.
    }

    private String getHost(FlowData data) {
        String reqHost = this.host;

        // Try and get the request host name so it can be used to request
        // the Json refresh in the JavaScript code.
        if (reqHost == null || reqHost.isEmpty()) {
            TryGetResult hostEvidence = data.tryGetEvidence(
                    Constants.EVIDENCE_HOST_KEY,
                    String.class);
            if (hostEvidence.hasValue()) {
                reqHost = hostEvidence.getValue();
            }
        }

        return reqHost;
    }

    private String getProtocol(FlowData data) {
        String reqProtocol = this.protocol;

        // Try and get the request protocol so it can be used to request
        // the JSON refresh in the JavaScript code.
        if (reqProtocol == null || reqProtocol.isEmpty()) {
            TryGetResult protocolEvidence = data.tryGetEvidence(
                    fiftyone.pipeline.core.Constants.EVIDENCE_PROTOCOL,
                    String.class);
            if (protocolEvidence.hasValue()) {
                reqProtocol = protocolEvidence.getValue();
            } else {
                // Couldn't get protocol from anywhere
                reqProtocol = Constants.DEFAULT_PROTOCOL;
            }
        }

        return reqProtocol;
    }

    private boolean getSupportsPromises(FlowData data) {
        boolean supportsPromises;

        // If device detection is enabled then try and get whether the
        // requesting browser supports promises. If not then default to false.
        try {
            AspectPropertyValue supportsPromisesValue =
                    data.getAs("Promise", AspectPropertyValue.class);
            supportsPromises = supportsPromisesValue.hasValue() &&
                    supportsPromisesValue.getValue() == "Full";
        } catch (PipelineDataException | PropertyMissingException e) {
            supportsPromises = false;
        }

        return supportsPromises;
    }

    private String getContextRoot(FlowData data) {
        String root = null;

        // Try and get the web server context root evidence so it can be
        // used to construct the correct path for the Json refresh.
        TryGetResult contextRoot = data.tryGetEvidence(
                EVIDENCE_WEB_CONTEXT_ROOT,
                String.class);
        if (contextRoot.hasValue()) {
            root = contextRoot.getValue();
        }
        return root;
    }

    private String getJsonObject(FlowData data) {
        String jsonObject;

        try {
            jsonObject = data.get(JsonBuilderData.class).getJson();
        } catch (PipelineDataException e){
            throw new PipelineConfigurationException("Json data is missing,"
                    + " make sure there is a JsonBuilder element before this"
                    + " JavaScriptBuilderElement in the pipeline", e);
        }

        return jsonObject;
    }

    private Map getParameters(FlowData data) throws UnsupportedEncodingException {
        HashMap parameters = new HashMap<>();

        Map queryEvidence = data
                .getEvidence()
                .asKeyMap();

        for(Map.Entry entry : queryEvidence.entrySet()){
            if(entry.getKey().startsWith(EVIDENCE_QUERY_PREFIX)){
                String key = entry.getKey().substring(entry.getKey().indexOf(EVIDENCE_SEPERATOR) + 1);
                key = URLEncoder.encode(key, "UTF-8");
                String value = URLEncoder.encode(entry.getValue().toString(), "UTF-8");
                parameters.put(key, value);
            }
        }

        return parameters;
    }

    private String getQueryParams(Map parameters) throws UnsupportedEncodingException {
        StringBuilder sb = new StringBuilder();

        Set keys = parameters.keySet();
        for (String key : keys) {
            sb.append(key);
            sb.append("=");
            sb.append(parameters.get(key));
            sb.append("&");
        }
        sb.deleteCharAt(sb.lastIndexOf("&"));

        return sb.toString();
    }

    private String serializeParameters(Map parameters) {
        JSONObject jsonObject = new JSONObject(parameters);
        return jsonObject.toString(0);
    }

    private String getUrl(String protocol, String host, String queryParams) {
        String url = null;
        if (protocol != null && !protocol.isEmpty() &&
                host != null && !host.isEmpty() &&
                endpoint != null && !endpoint.isEmpty()) {
            boolean contextRootPopulated = contextRoot != null &&
                    !contextRoot.isEmpty() && !contextRoot.equals("/");

            // Make sure that each part of the URL except host starts with a '/'
            // and each part except endpoint does NOT end with one.
            if (endpoint.charAt(0) != '/') {
                endpoint = "/" + endpoint;
            }
            if (contextRootPopulated && contextRoot.charAt(0) != '/') {
                contextRoot = "/" + contextRoot;
            }
            if (host.charAt(host.length() - 1) == '/') {
                host = host.substring(0, host.length() - 1);
            }
            if (contextRootPopulated && contextRoot.charAt(contextRoot.length() - 1) == '/') {
                contextRoot = contextRoot.substring(0, contextRoot.length() - 1);
            }

            url = protocol + "://" + host +
                    (contextRootPopulated ? contextRoot : "") +
                    endpoint +
                    (queryParams.isEmpty() ? "" : "?" + queryParams);
        }

        return url;
    }

    private String getSessionId(FlowData data) {
        String sessionId = "";
        TryGetResult trySessionId = data.tryGetEvidence(EVIDENCE_SESSIONID, String.class);

        if (trySessionId.hasValue()) {
            sessionId = trySessionId.getValue();
        }

        return sessionId;
    }

    private Integer getSequence(FlowData data) {
        Integer sequence = 1;
        TryGetResult trySequence = data.tryGetEvidence(EVIDENCE_SEQUENCE, Integer.class);
        if (trySequence.hasValue()) {
            sequence = trySequence.getValue();
        }

        return sequence;
    }

    private void buildJavaScript(
        FlowData data,
        String jsonObject,
        boolean supportsPromises,
        String url,
        String parameters,
        String sessionId,
        int sequence) {
        JavaScriptBuilderDataInternal elementData =
            (JavaScriptBuilderDataInternal)data.getOrAdd(
                getElementDataKey(),
                getDataFactory());

        String objectName;
        // Try and get the requested object name from evidence.
        TryGetResult res = data.tryGetEvidence(
            EVIDENCE_OBJECT_NAME,
            String.class );
        if (res.hasValue() == false ||
            res.getValue().isEmpty()) {
            objectName = objName;
        } else {
            objectName = res.getValue();
        }

        boolean cookies;
        // Try and get the requested enable cookies from evidence.
        TryGetResult cookieVal = data.tryGetEvidence(
            EVIDENCE_ENABLE_COOKIES,
            String.class);
        if (cookieVal.hasValue() == false ||
            cookieVal.getValue().isEmpty()) {
            cookies = enableCookies;
        } else {
            cookies = Boolean.parseBoolean(cookieVal.getValue());
        }

        boolean updateEnabled = url != null && url.isEmpty() == false;

        // This check won't be 100% fool-proof but it only needs to be
        // reasonably accurate and not take too long.
        boolean hasDelayedProperties = jsonObject != null &&
            jsonObject.contains("delayexecution");

        JavaScriptResource javaScriptObj = new JavaScriptResource(
            objectName,
            jsonObject,
            sessionId,
            sequence,
            supportsPromises,
            url,
            parameters,
            cookies,
            updateEnabled,
            hasDelayedProperties);
      
        StringWriter stringWriter = new StringWriter();
        mustache.execute(stringWriter, javaScriptObj.asMap());

        String content = stringWriter.toString();

        elementData.setJavaScript(content);
    }
}
//! [class]




© 2015 - 2025 Weber Informatics LLC | Privacy Policy