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

org.jolokia.client.request.J4pRequest Maven / Gradle / Ivy

The newest version!
package org.jolokia.client.request;

/*
 * Copyright 2009-2013 Roland Huss
 *
 * 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.
 */

import java.lang.reflect.Array;
import java.time.temporal.ChronoField;
import java.time.temporal.Temporal;
import java.util.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import org.jolokia.json.*;

/**
 * Request object abstracting a request to a j4p agent.
 *
 * @author roland
 * @since Apr 24, 2010
 */
public abstract class J4pRequest {


    // request type
    private final J4pType type;

    // "GET" or "POST"
    private String preferredHttpMethod;

    // target configuration for this request when used as a JSR-160 proxy
    private final J4pTargetConfig targetConfig;

    /**
     * Constructor for subclasses
     * @param pType type of this request
     * @param pTargetConfig a target configuration if used in proxy mode or null
     *                      if this is a direct request
     */
    protected J4pRequest(J4pType pType, J4pTargetConfig pTargetConfig) {
        type = pType;
        targetConfig = pTargetConfig;
    }

    /**
     * Escape a input (like the part of an path) so that it can be safely used
     * e.g. as a path
     *
     * @param pInput input to escape
     * @return the escaped input
     */
    public static String escape(String pInput) {
        return pInput.replaceAll("!","!!").replaceAll("/","!/");
    }

    /**
     * Get the type of the request
     *
     * @return request's type
     */
    public J4pType getType() {
        return type;
    }

    /**
     * Get a target configuration for use with an agent in JSR-160 proxy mode
     *
     * @return the target config or null if this is a direct request
     */
    public J4pTargetConfig getTargetConfig() {
        return targetConfig;
    }

    /**
     * The preferred HTTP method to use (either 'GET' or 'POST')
     * @return the HTTP method to use for this request, or null if the method should be automatically selected.
     */
    public String getPreferredHttpMethod() {
        return preferredHttpMethod;
    }

    /**
     * Set the preferred HTTP method, either 'GET' or 'POST'.
     *
     * @param pPreferredHttpMethod HTTP method to use.
     */
    public void setPreferredHttpMethod(String pPreferredHttpMethod) {
        preferredHttpMethod = pPreferredHttpMethod != null ? pPreferredHttpMethod.toUpperCase() : null;
    }

    // ==================================================================================================
    // Methods used for building up HTTP Requests and setting up the reponse
    // These methods are package visible only since are used only internally

    // Get the parts to build up a GET url (without the type as the first part)
    abstract List getRequestParts();

    // Get a JSON representation of this request
    JSONObject toJson() {
        JSONObject ret = new JSONObject();
        ret.put("type", type.name());
        if (targetConfig != null) {
            ret.put("target", targetConfig.toJson());
        }
        return ret;
    }

    /**
     * Create a response from a given JSON response
     *
     * @param pResponse http response as obtained from the Http-Request
     * @return the create response
     */
    abstract > R createResponse(JSONObject pResponse);

    // Helper class
    protected void addPath(List pParts, String pPath) {
        if (pPath != null) {
            // Split up path
            pParts.addAll(splitPath(pPath));
        }
    }

    /**
     * Serialize an object to a string which can be uses as URL part in a GET request
     * when object should be transmitted to the agent. The serialization is
     * rather limited: If it is an array, the array's member's string representation are used
     * in a comma separated list (without escaping so far, so the strings must not contain any
     * commas themselves). If it is not an array, the string representation ist used (Object.toString())
     * Any null value is transformed in the special marker [null] which on the
     * agent side is converted back into a null.
     * 

* You should consider POST requests when you need a more sophisticated JSON serialization. *

* @param pArg the argument to serialize for an GET request * @return the string representation */ protected String serializeArgumentToRequestPart(Object pArg) { if (pArg != null) { if (pArg.getClass().isArray()) { return getArrayForArgument((Object[]) pArg); } else if (List.class.isAssignableFrom(pArg.getClass())) { List list = (List) pArg; Object[] args = new Object[list.size()]; int i = 0; for (Object e : list) { args[i++] = e; } return getArrayForArgument(args); } else if (Date.class == pArg.getClass()) { // pass Date as long (there's no TZ information here just like in java.time.Instant) Date d = (Date) pArg; return Long.toString(d.getTime()); } else if (Temporal.class.isAssignableFrom(pArg.getClass())) { // special handling for the temporals that can easily be converted to unix time (in nanos) Temporal t = (Temporal) pArg; if (t.isSupported(ChronoField.INSTANT_SECONDS)) { long instant = t.getLong(ChronoField.INSTANT_SECONDS) * 1_000_000_000L + t.getLong(ChronoField.NANO_OF_SECOND); return Long.toString(instant); } else { // for now we can't nicely convert it and we don't know what's the pattern used // at server side return t.toString(); } } } return nullEscape(pArg); } /** * Serialize an object to an string or JSON structure for write/exec POST requests. * Serialization is up to now rather limited: *
    *
  • * If the argument is null null is returned. *
  • *
  • * If the argument is of type {@link org.jolokia.json.JSONStructure}, the it is used directly for inclusion * in the POST request. *
  • *
  • * If the argument is an array, this array's content is put into * an {@link org.jolokia.json.JSONArray}, where each array member is serialized recursively. *
  • *
  • * If the argument is a map, it is transformed into a {@link org.jolokia.json.JSONObject} with the keys taken * directly from the map and the values recursively serialized to their JSON representation. * So it is only save fto use or a simple map with string keys. *
  • *
  • * If the argument is a {@link Collection}, it is transformed into a {@see JSONArray} with * the values recursively serialized to their JSON representation. *
  • *
  • * Otherwise the object is used directly. *
  • *
* * Future version of this lib will probably provide a more sophisticated serialization mechanism. * This is how it is supposed to be for the next release, currently a simplified serialization is in place * * @param pArg the object to serialize * @return a JSON serialized object */ protected Object serializeArgumentToJson(Object pArg) { if (pArg == null) { return null; } else if (pArg instanceof JSONStructure) { return pArg; } else if (pArg.getClass().isArray()) { return serializeArray(pArg); } else if (pArg instanceof Map) { //noinspection unchecked return serializeMap((Map) pArg); } else if (pArg instanceof Collection) { return serializeCollection((Collection) pArg); } else if (Date.class == pArg.getClass()) { // pass Date as long (there's no TZ information here just like in java.time.Instant) Date d = (Date) pArg; return Long.toString(d.getTime()); } else if (pArg instanceof Temporal) { // special handling for the temporals that can easily be converted to unix time (in nanos) Temporal t = (Temporal) pArg; if (t.isSupported(ChronoField.INSTANT_SECONDS)) { return t.getLong(ChronoField.INSTANT_SECONDS) * 1_000_000_000L + t.getLong(ChronoField.NANO_OF_SECOND); } else { // for now we can't nicely convert it and we don't know what's the pattern used // at server side return t.toString(); } } else { return pArg instanceof Number || pArg instanceof Boolean ? pArg : pArg.toString(); } } // pattern used for escaping business private static final Pattern SLASH_ESCAPE_PATTERN = Pattern.compile("((?:[^!/]|!.)*)(?:/|$)"); private static final Pattern UNESCAPE_PATTERN = Pattern.compile("!(.)"); /** * Split up a path taking into account proper escaping (as described in the * reference manual). * * @param pArg string to split with escaping taken into account * @return split element or null if the argument was null. */ protected List splitPath(String pArg) { List ret = new ArrayList<>(); if (pArg != null) { Matcher m = SLASH_ESCAPE_PATTERN.matcher(pArg); while (m.find() && m.start(1) != pArg.length()) { ret.add(UNESCAPE_PATTERN.matcher(m.group(1)).replaceAll("$1")); } } return ret; } // ===================================================================================================== private Object serializeCollection(Collection pArg) { JSONArray array = new JSONArray(pArg.size()); for (Object value : pArg) { array.add(serializeArgumentToJson(value)); } return array; } private Object serializeMap(Map pArg) { JSONObject map = new JSONObject(); for (Map.Entry entry : pArg.entrySet()) { map.put(entry.getKey(), serializeArgumentToJson(entry.getValue())); } return map; } private Object serializeArray(Object pArg) { int length = Array.getLength(pArg); JSONArray innerArray = new JSONArray(length); for (int i = 0; i < length; i++ ) { innerArray.add(serializeArgumentToJson(Array.get(pArg, i))); } return innerArray; } private String getArrayForArgument(Object[] pArg) { StringBuilder inner = new StringBuilder(); for (int i = 0; i< pArg.length; i++) { inner.append(nullEscape(pArg[i])); if (i < pArg.length - 1) { inner.append(","); } } return inner.toString(); } // null escape used for GET requests private String nullEscape(Object pArg) { if (pArg == null) { return "[null]"; } else if (pArg instanceof String && ((String) pArg).isEmpty()) { return "\"\""; } else if (pArg instanceof JSONStructure) { return ((JSONStructure) pArg).toJSONString(); } else { return pArg.toString(); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy