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

com.unowmo.microwrap.MultiEndpointApi Maven / Gradle / Ivy

package com.unowmo.microwrap;

import java.util.*;
import java.util.concurrent.*;
import java.io.*;
import com.fasterxml.jackson.core.*;
import com.fasterxml.jackson.databind.*;
import com.fasterxml.jackson.databind.annotation.*;
import com.amazonaws.services.lambda.runtime.*;

/**
 * AWS Lambda handler implementation that decodes deserialized requests and
 * invokes the logic block associated with that command.
 * 
 * @author Kirk Bulis
 *
 */
public abstract class MultiEndpointApi, R extends MultiEndpointApi.WrappedResources> implements RequestStreamHandler {
    private final Handled [] hooks; 

    /**
     * Base container for implementations to wrap request handling with resource
     * allocation and lifetime management handling.
     */
    public static abstract class ResourceWrapping implements AutoCloseable {

    	public abstract void onCommit(final T context, final String command, final String trusted, final Date started);

    }

    /**
     * Base container implementation for holding onto request-specific values
     * and facilities as part of normal request handling. Container should
     * define a default constructor. 
     */
    public static class ContainerContext {

        public Tracer logger = null;
        public String detail = null;

    }

    /**
     * Base container implementation for holding onto wrapped resources alive
     * during request processing and managed by the wrapper. 
     */
    public static class WrappedResources {

    }

    /**
     * Simple logger wrapper for tracing activity during request processing.
     */
    public static class Tracer {
        private final LambdaLogger delegate;

        public Tracer log(final String message) {
            if (this.delegate != null)
            {
                this.delegate.log("(" + TimeUnit.NANOSECONDS.toMillis(System.nanoTime()) + ") " + (message != null ? message : "null"));
            }

            return this;
        }
    
        private Tracer(final LambdaLogger delegate) {
            this.delegate = delegate;
        }

    }

    /**
     * Event for container to initialize context details using given environment
     * parameters and facilities.
     * 
     * @param context micro framework facility context
     * @param command request command to process
     * @param trusted request security token
     * @param region location hint for services
     * @param config execution configuration
     * @param logger logging facility
     * @return initialized container context instance
     * @throws IOException raised on any error
     */
    protected abstract T prepareRequestContainer(final Context context, final String command, final String trusted, final String region, final String config, final Tracer logger) throws IOException;

    /**
     * Event for container to optionally allocate a resource wrapper of request
     * handling.
     * 
     * @param containerContext context for allocation
     * @throws IOException raised on any error
     * @return new wrapper for request cycle management
     */
    protected abstract W allocateResourceWrapper(final T containerContext) throws IOException;

    /**
     * Event for container to optionally allocate a wrapped resource of request
     * handling.
     * 
     * @param containerContext context for allocation
     * @param resourceWrapper initializer/manager of wrapped resources
     * @throws IOException raised on any error
     * @return new wrapper for request cycle management
     */
    protected abstract R allocateWrappedResource(final T containerContext, final W resourceWrapper) throws IOException;
    
    /**
     * Connects and initializes shared context parameters as part of specialized
     * request processing at different layers. Implementers must call the super
     * form before returning.
     * 
     * @param context container context being updated
     * @param command request command to process
     * @param trusted request security token
     * @param region location hint for services
     * @param config execution configuration
     * @param logger logging facility
     */
    protected void fixupRequestContainer(final T context, final String command, final String trusted, final String region, final String config, final Tracer logger) {
        context.detail = String.format
            ( "Processing '%s' with token '%s' in " + region + " as " + config
            , command
            , trusted
            );
        
        context.logger = logger;
    }

    /**
     * Actual custom lambda handler hook.
     * 
     * @param source request body streamed
     * @param target response body output
     * @param context execution context
     */
    @Override
    public final void handleRequest(final InputStream source, final OutputStream target, final Context context) {
        final Date started = new Date();

        try
        {
            // Looks up region and config specified through calling context
            // provided by execution container. Though not platform specific,
            // the use of both was born from aws convention.

            String config = "";
            String region = "";

            if (context != null)
            {
                if (context.getInvokedFunctionArn() != null)
                {
                    String [] arn = context.getInvokedFunctionArn().split(":");
                    
                    if (arn.length > 3)
                    {
                        region = arn[3].trim();
                        
                        config = "prod";
                    }
                }
                
                if (context.getClientContext() != null)
                {
                    region = context.getClientContext().getEnvironment().getOrDefault
                        ( "msRegion"
                        , region
                        );

                    config = context.getClientContext().getEnvironment().getOrDefault
                        ( "msConfig"
                        , config
                        );
                }
            }

            if (region.equalsIgnoreCase("") == true)
            {
                throw new IOException
                    ( "Handler request context lacks valid region"
                    );
            }
            
            if (config.equalsIgnoreCase("") == true)
            {
                throw new IOException
                    ( "Handler request context lacks valid config"
                    );
            }

            // Execute actual implementation and serialize result for calling
            // container to consume. Expects json, which is another convention
            // related to aws implementation, but general enough to prove
            // useful in a wide range of platforms.
            //
            // We look for matching request command handlers in the order
            // given on construction. This can be optimized, but we don't
            // really expect a large set.

            String requesting = "";
            String responseOf = "";

            if (source == null)
            {
                throw new IOException
                    ( "Invalid command request source; expecting body with command parameter"
                    );
            }
            
            try (final InputStreamReader reader = new InputStreamReader(source, "utf-8"))
            {
                final char buffer[] = new char[16 * 1024];

                while (true)
                {
                    int n = reader.read(buffer);
                    
                    if (n > 0)
                    {
                        requesting += new String(buffer, 0, n);
                    }
                    else
                    if (n < 0)
                    {
                        break;
                    }
                }
            }
            
            try
            {
                Posting deserialized = mapper.readValue(requesting, Posting.class);
                Returns answer;

                answer = new Returns
                    ( String.format
                        ( "command request '%s' not supported"
                        , deserialized.command
                        )
                    );
                
                if (deserialized.command.equalsIgnoreCase("getappdetail") == false)
                {
                    Tracer logger = new Tracer(context != null ? context.getLogger() : null);

                    try
                    {
                        final T containerContext = this.prepareRequestContainer(context, deserialized.command, deserialized.trusted, region, config, logger);

                        this.fixupRequestContainer
                            ( containerContext
                            , deserialized.command
                            , deserialized.trusted
                            , region
                            , config
                            , logger
                            );

                        containerContext.logger.log
                            ( "running '" + deserialized.command + "' with request = " + deserialized.request
                            );
                        
                        for (final Handled handled : this.hooks)
                        {
                            if (handled != null && handled.command.equalsIgnoreCase(deserialized.command) == true)
                            {
                            	try (final W wrapper = this.allocateResourceWrapper(containerContext))
                            	{
                            		final R wrapped = this.allocateWrappedResource(containerContext, wrapper);

                            		if (wrapped != null)
                            		{
		                                Object object = handled.doCommand(containerContext, wrapped, deserialized.request.toString(), started);
		                                
		                                if (object != null)
		                                {
		                                    containerContext.logger.log
		                                        ( "success"
		                                        );
		
		                                    answer = new Returns
		                                        ( "success"
		                                        , object
		                                        );
		                                }
		                                else
		                                {
		                                    throw new IOException
		                                        ( "Handler logic response came back null"
		                                        );
		                                }
		                                
		                                wrapper.onCommit
		                                	( containerContext
		                                	, deserialized.command
		                                	, deserialized.trusted
		                                	, started
		                                	);
                                	}
                            	}
                            	
                            	break;
                            }
                        }
                    }
                    catch (IOException eX)
                    {
                        throw eX;
                    }
                    catch (Exception eX)
                    {
                        throw new IOException
                            ( "Unable to initialize container and execute command"
                            , eX
                            );
                    }
                }
                else
                {
                    try (final InputStream declared = this.getClass().getClassLoader().getResourceAsStream("app.properties"))
                    {
                        if (declared != null)
                        {
                            Properties properties = new Properties();
    
                            properties.load
                                ( declared 
                                );
                            
                            answer = new Returns
                                ( "success"
                                , new Version
                                    ( properties.getProperty("detail.name")
                                    , properties.getProperty("detail.version")
                                    )
                                );
                        }
                        else
                        {
                            answer = new Returns
                                ( "success"
                                , new Version
                                    ( "unknown"
                                    , "0.0.0"
                                    )
                                );
                        }
                    }
                    catch (Exception eX)
                    {
                        answer = new Returns
                            ( String.format
                                ( "failed%s"
                                , eX.getMessage() != null ? " because " + eX.getMessage().toLowerCase() : ""
                                )
                            );
                    }
                }
                
                responseOf = answer != null ? mapper.writeValueAsString(answer) : "{ }";
            }
            catch (Exception eX)
            {
                throw eX;
            }

            // Now writing whatever result was obtained as a serialized blob
            // with replacements for operational parameters if the container
            // cares to inject them.
            //
            // Note that a failure to write could throw out and subsequently
            // append the exception details.
            
            try
            {
                final Date wrapped = new Date();

                responseOf = responseOf.replace("((running))", "" + (wrapped.getTime() - started.getTime()));
                responseOf = responseOf.replace("((execute))", "" + (context.getAwsRequestId()));
                responseOf = responseOf.replace("((started))", "" + (started.getTime()));
                responseOf = responseOf.replace("((wrapped))", "" + (wrapped.getTime()));
                
                target.write
                    ( responseOf.replace("''",  "\"").getBytes("utf8")
                    );
            }
            catch (Exception eX)
            {
                throw eX;
            }

            return;
        }
        catch (Exception eX)
        {
            // Oops. Handle error and let's figure this out.
    
            try
            {
                target.write
                    ( String.format
                        ( "{ ''results'': ''Failed%s'', ''started'': %d, ''execute'': ''%s'' }"
                        , eX.getMessage() != null ? " because " + eX.getMessage().toLowerCase().replace('\'', '`') : ""
                        , started.getTime()
                        , context.getAwsRequestId()
                        ).replace("''",  "\"").getBytes("utf8")
                    );
            }
            catch (Exception nX)
            {
            }

            if (context != null)
            {
                context.getLogger().log
                    ( String.format
                        ( "failed%s"
                        , eX.getMessage() != null ? " because " + eX.getMessage().toLowerCase() : ""
                        )
                    );
            }
        }
    }
    
    /**
     * Base interface for all handler hooks to process requests. 
     */
    public static abstract class Handled {
        final String command;
        
        public abstract Object doCommand(final T context, final R wrapped, final String posting, final Date started) throws IOException;

        public Handled(final String commandLabel) {
            this.command = commandLabel;
        }

    }

    /**
     * Container for processing. 
     */
	@JsonDeserialize(using=Posting.Deserializer.class)
    public static class Posting {

        public String command = "";
        public String trusted = "";
        public String request = "";

		public static class Deserializer extends JsonDeserializer {

			@Override
		    public Posting deserialize(JsonParser parser, DeserializationContext context) throws IOException, JsonProcessingException {
		    	Posting that = new Posting();
		    	
		    	try
		    	{
		    		JsonNode node = parser.getCodec().readTree(parser);
		    		
		    		if (node.get("command") != null)
		    		{
		    			that.command = node.get("command").asText("?").trim();
		    		}

		    		if (node.get("trusted") != null)
		    		{
		    			that.trusted = node.get("trusted").asText("").trim();
		    		}

		    		if (node.get("request") != null)
		    		{
		    			that.request = node.get("request").toString().trim();
		    		}
		    	}
		    	catch (JsonProcessingException eX)
		    	{
		    		throw eX;
		    	}
		    	catch (IOException eX)
		    	{
		    		throw eX;
		    	}
		    	
		    	return that;
		    }

		}
        
    }

    /**
     * Container for processing. 
     */
    public static class Returns {

        public String results = "";
        public Object o = null;
        
        public Returns(final String results, final Object o) {
            this.results = results != null ? results.replace('"', '\'') : "";
            this.o = o;
        }

        public Returns(final String results) {
            this.results = results != null ? results.replace('"', '\'') : "";
        }

        final String running = "((running))";
        final String execute = "((execute))";
        
    }

    /**
     * Container for processing. 
     */
    public static class Version {

        public final String name;
        public final String version;

        public Version(final String name, final String version) {
            this.name = name;
            this.version = version;
        }

    }

    /**
     * Container for processing.
     */
    public static class Options {
    	
    	public String region = "";
    }

    /**
     * Construct default.
     * 
     * @param hooks handler endpoint implementations provided by container
     */
    protected MultiEndpointApi(final Handled [] hooks) {
        this.hooks = hooks;
    }

    /**
     * Shared facility.
     */
    public static ObjectMapper mapper = new ObjectMapper();
    static {
		mapper.configure
			( DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES
			, false
			);

		mapper.configure
			( SerializationFeature.FAIL_ON_EMPTY_BEANS
			, false
			);
    };

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy