Hedge is a platform agnostic ClojureScript framework for deploying ring compatible handlers to various environments with focus on serverless deployments.
- /library contains wrapping and transformation code for adapting handlers to various platforms
- /boot contains Hedge specific Boot tasks for code generation
- /acceptance contains Concordion based acceptance test suite which also works as technical documentation on how to use Hedge
- asynchronous ring handlers not yet supported
Copyright © 2016-2018 Siili Solutions Plc.
Distributed under the Eclipse Public License 1.0.
- Forewords
- Preparations and Required Software
- How Hedge Works
- Preparing Hedge Authentication Against Azure
- Authentication for AWS
- Supported Handler Types
- Input/Output Bindings
- Logging
- Basic Serverless Function Project Structure
- hedge.edn
- Handler signatures
- Exception Handling inside Serverless Function
- Testing
- Deploying to Azure
- Deploying to AWS
- Other Usage Examples
This document gets you started writing and deploying serverless functions to different environments. Hedge is under development, so things can change.
- Java JDK 8 - Boot runs on JVM
- Boot-CLJ - Required to build ClojureScript projects and use Hedge
- Node.js - Required for running unit tests
- Azure CLI - (Optional) Can be used when creating Service Principal for Azure and managing Azure resources
- AWS CLI - (Optional) Can be used to create credentials and to remove old test stacks from AWS
ClojureScript is compiled and optimized into JavaScript than can be run on Node.js or similar runtime environment available in serverless environments. Hedge takes care of generating code that is compatible and runnable on the deployment target. Hedge can then use provided authentication profile and deploy the compiled code to the serverless environment.
Requirements: You must be subscription owner to be able to create service principals, or ask a subscription owner to generate the principal for you. This document describes how you can create service principal with Azure CLI from the command line.
If you haven't already performed log in with Azure CLI, do it and follow on-screen instructions.
az login
Make sure you are using correct subscription
az account list --output table
Optionally change the subscription
az account set --subscription "<your subscription name>"
Example: Create Service Principal with the name "MyNameServicePrincipal"
az ad sp create-for-rbac --name "MyNameServicePrincipal" --sdk-auth true > MyNameServicePrincipal.json
Hint: Use a meaningful name that you can identify, so you can find it later if you need to remove it. Keep the generated file in a secure place, because it contains contributor role credentials by default that are able to affect things in your whole subscription.
Configure your environment
export AZURE_AUTH_LOCATION=/path-to-your/MyNameServicePrincipal.json
Easiest way to create credentials for AWS to follow AWS guide for their CLI.
- Create new access key and secret access key in AWS IAM console
- Setup credentials locally
- By running command
aws configure
or - By creating
~/.aws/credentials
file manually
Note: If you for some reason need to change authentication profile name you can use environment variable AWS_PROFILE
with
name of the profile you wish to use.
If you don't want to save access key and secret access key into disk they can be
passed to hedge by using environment variable AWS_ACCESS_KEY_ID
and
AWS_SECRET_ACCESS_KEY
To store environment variables for current shell session use command
export <VARIABLE_NAME-1>=<VALUE_1> <VARIABLE_NAME-2>=<VALUE_2>
Handler type | Trigger | Trigger Input | Output | Azure | AWS |
---|---|---|---|---|---|
:api | HTTP Request | HTTP Request | HTTP Response | Yes | Yes |
:timer | Cron schedule | Timer | result->JS | Yes | Yes |
:queue | New message in queue | Message | result->JS | Yes | Yes |
Other handler types are planned.
Notes on Azure handlers: In Azure you can configure the output to be passed as return value to other Azure services (i.e. queue, table storage etc).
Hedge can abstract Inputs and Outputs similar to Azure. A function can have a variable amount of input and output bindings. Inputs are passed to the function on invocation and outputs are persisted on function successful complete (unless nil). Depending on configuration, you can use different implementations (if available) with same abstraction.
type | Input | Output | Azure | AWS |
---|---|---|---|---|
:queue | n/a | Queue | Storage Queue | TBA |
:queue | n/a | Queue | ServiceBus Queue | TBA |
:queue | n/a | Queue | ServiceBus Topic | TBA |
:table | Key-Value Store | Key-Value Store | Table Storage | TBA |
:db | Database | Database | CosmosDB | TBA |
Logging inside your serverless function is dependent on target platform. The Hedge handlers require internally Timbre as logging system. You can use Timbre in your handler with following require:
Example on handler that uses Timbre logging (in a file src/my-cool-function/core.cljs
)
(ns my-cool-function.core
(:require [taoensso.timbre :as timbre
:refer [log trace debug info warn error fatal report
logf tracef debugf infof warnf errorf fatalf reportf
spy get-env log-env]]))
; default logging level is :debug
(timbre/set-level! :trace)
(defn handler [req]
(info "hello info level")
(error "hello error level"))
A basic structure can be found in one of the examples in AWS example repository.
Note: Example directories might be merged to master repo at some point.
- src/ - This directory contains your CLJS source files
- target/ - Optionally persisted compiled output directory
- resources/hedge.edn - Includes configuration information that maps your program function to the serverless function entry point. You can configure here if a function is protected with authentication code or available without authentication. Also function type is defined here (api, timer, ...)
- test/ - This directory contains your unit tests
- boot.properties - Boot properties
- build.boot - Your build configuration file
- package.json - Optionally, if you use external Node modules
- node_modules - Optionally, if you have installed Node modules
Refer example repositories for more info.
build.boot is boot configuration file. Hedge commands are implemented as boot task.
Example configuration with Hedge and boot-cljs-test:
(set-env! :source-paths #{"src"}
:resource-paths #{"resources"}
:dependencies '[[org.clojure/clojurescript "1.9.946"]
[adzerk/boot-cljs "1.7.228-2" :scope "test"]
[crisptrutski/boot-cljs-test "0.3.4" :scope "test"]
[siili/boot-hedge "0.1.0" :scope "test"]
[siili/hedge "0.1.0"]])
; refer example repository for detailed
(require '[boot-hedge.core :as hedge])
(hedge/hedge-init!)
(require '[crisptrutski.boot-cljs-test :refer [test-cljs report-errors!] :as cljs-test])
(deftask testing [] (set-env! :source-paths #(conj % "test")) identity)
(ns-unmap 'boot.user 'test)
(deftask test []
(comp (testing)
(test-cljs :js-env :node
:exit? true)))
resources/hedge.edn contains the configuration for the given functions.
Example of a configuration for two apis, two timers and one queue triggered function:
{:api {"api1" {:handler my_cool_function.core/crunch-my-data :authorization :anonymous}
"api2" {:handler my_cool_function.core/hello :authorization :function}}
:timer {"timer1" {:handler my_cool_function.core/timer-handler :cron "*/10 * * * *"}
"timer2" {:handler my_cool_function.core/timer-handler-broken :cron "*/10 * * * *"}}
:queue {:"queue" {:handler my_cool_function.core/timer-handler
:queue "nameOfQueue"
:connection "environmental variable that holds the connection string (Azure)"}}}
Common structure:
{:handler-type {"name-given-to-api" {:handler namespace.core/name-of-function}}}
Azure :api specific:
:authorization :anonymous
or :authorization :function
- defines if the HTTP endpoint public or access key protected
:timer specific:
:cron "{minutes} {hours} {day-of-month} {months} {day-of-week}"
Example on cron expression that will trigger every minute: "*/1 * * * *"
Note on used cron expression:
- Only simple expressions are supported
- *, numbers and / wildcard
- setting both day-of-month and day-of-week is not supported (AWS limitation)
- names of days/months and L, W, ? and # wildcards are not supported
- Azure has a field for seconds and AWS doesn't (field results to 0 when generating Azure function.json)
- AWS has a field for years and Azure doesn't (field results to *)
Azure users: You can modify the resulted function.json if you need to incorporate seconds in your cron expression
Azure :queue specific:
:accessRights "Manage"
or :accessRights "Listen"
- if defined, will use servicebus queue/topic queue instead of storage queue
:connection "ENV_VARIABLE"
- function app environmental variable that holds the connection string to the queue service (for example AzureWebJobsStorage for storage queues)
:subscription "subscriptionName"
- if defined together with :accessRights
, will use a service bus topic & subscription. Subscription will be created if it does not exist.
If your function throws an unhandled exception or fails, the message is returned to the queue, retried and finally put into poison queue (azure storage queue) or dead-letter queue (servicebus queue).
Note: It might be that it is not possible to put a servicebus topic message to dead-letter queue (when your function fails), because design decisions in Azure function runtime.
Storage Queue polling frequency and servicebus queue settings for the function app runtime can be set in host.json, see https://docs.microsoft.com/en-us/azure/azure-functions/functions-host-json#queues
Please note you must create the queues/topics yourself (you can do it through the portal or use Azure Storage Explorer with storage Queues).
Define Function Inputs and Outputs in hedge.edn:
Please note :connection is Azure specific Function App Setting (env variable) that contains the connection string. In Azure you can specify target Queue implementation, see below.
{:api {"api1" {:handler my_cool_function.core/crunch-my-data :authorization :anonymous}
:inputs [{:type :table
:key "in1"
:name "inputTable"
:connection "AzureWebJobStorage"}
{:type :db
:key "in2"
:name "inputDb"
:collection "collection"
:connection "CosmosDBConnection"}]
:outputs [{:type :queue
:key "out1"
:name "queue"
:connection "AzureWebJobsStorage"}
{:type :queue
:key "out2"
:accessRights "Manage"
:name "queue"
:connection "SBQueueConnection"}
{:type :queue
:key "out3"
:topic true
:accessRights "Manage"
:name "queue"
:connection "SBTopicConnection"}
{:type :db
:key "out4"
:name "db"
:collection "collection"
:connection "ConnectionString"}
{:type :table
:key "out5"
:name "table"
:connection "ConnectionString"}]
Short examples on handler signatures:
(defn api-handler
"req contains the incoming http request, return value is passed as the response"
[req]
"Hello World!")
(defn timer-handler
"timer contains the scheduled trigger timestamp, return can be passed to function output i.e. other wired service"
[timer]
"Hello World!")
(defn queue-handler
"message contains the message payload, return can be passed to function output i.e. other wired service"
[message]
"Hello World!)
If you are using inputs and outputs, you can currently use any of the following signature additions on any type of handler:
(defn api-handler-with-inputs
"req contains the incoming http request, inputs contains a map of given inputs described in hedge.edn"
[req & {:keys [inputs]}]
(info
"Read table from table storage: "
(-> inputs :in1))
"This goes as return value to HTTP response")
(defn api-handler-with-outputs
"req contains the incoming http request, outputs contain a map of atoms for given outputs defined in hedge.edn"
[req & {:keys [outputs]}]
; write two messages to storage queue (mapped as :out1)
(reset! (-> outputs :out1 :value) [{:message {:id "1" :content "hello world"}} {:message {:id "2" :content "hello again"}}])
; one message to servicebus queue (Azure only)
(reset! (-> outputs :out2 :value) {:message {:id "1" :content "hello world"}})
; two messages to to servicebus topic (Azure only)
(reset! (-> outputs :out3 :value) [{:message {:id "1" :content "hello world"}} {:message {:id "2" :content "hello again"}}])
; to db (example with two rows)
(reset! (-> outputs :out4 :value) [{:name "John Doe"
:address "123 Hollywood"}
{:name "Jane Doe"
:address "123 Hollywood"
:info "available}])
; write row to table storage (mapped as :out5)
(reset! (-> outputs :out5 :value) {:PartitionKey "partitionkey-1" :RowKey "rowkey-1 :value "A value stored."})
"This goes as return value to HTTP response")
(defn api-handler-with-inputs-and-outputs
[req & {:keys [inputs outputs]}]
; as above
"This goes as return value to HTTP response")
See examples for more usage patterns.
These instructions especially apply for Azure, and is considered as best practice for AWS also in the context of Hedge.
If your function throws an uncaught exception, it will be passed as Error to the serverless runtime. This will be visible in Application Insights logs as user reported execution failure. This is useful when for example receiving messages from a queue but your function would like to abort processing on failure and return the message to queue.
If you are using go
blocks (i.e. performing async HTTP requests), throwing an uncaught exception (either accidentally or signaling the runtime on purpose) inside a go
block will not be caught by the runtime and unexpected behavior can be expected. The solution is to use go-try
:
Add dependency to your functions build.boot [org.clojars.akiel/async-error "0.2"]
see https://github.com/alexanderkiel/async-error
Require the necessary macros in your handler [async-error.core :refer-macros [go-try <?]]
.
Instead of using go
and <!
, use go-try
and <?
. Thus exceptions from port reading will be passed in the channels upstream and finally to the serverless runtime.
Testing your function (unit testing) requires a JavaScript runtime, ie. Node.js (or PhantomJS) because ClojureScript is compiled to JavaScript and needs a platform to run on.
To run unit tests in test/ -folder
boot test
To run unit tests when files change in your project (watch project)
boot watch test
To deploy, Hedge requires that you create your Resource Group. Hedge can create a Storage Account and Function App for you, but we recommend that you create these by your self so that you can choose where you host your serverless function code. One Function App can contain multiple serverless functions.
When you create these by your self you can choose location of data center, storage options and service plan.
Storage account is required to store your function code and logs that your functions create and other settings.
You can do it from the Azure Portal or use Azure CLI (Instructions below).
Example create the Resource Group in northeurope data center:
az group create --name NameOfResourceGroup --location northeurope
To create your storage account with simplest storage option in north europe:
az storage account create --name NameOfStorageAccount --location northeurope --resource-group NameOfResourceGroup --sku Standard_LRS
To create your function app with consumption plan (Windows Server backed serverless runtime with dynamic resource scaling, no dedicated VM):
az functionapp create --name NameOfFunctionApp --storage-account NameOfStorageAccount --resource-group NameOfResourceGroup --consumption-plan-location northeurope
To compile and deploy your function to Azure:
boot azure/deploy -a NameOfFunctionApp -r NameOfResourceGroup
If your authentication file is correctly generated and found in the environment, your function should deploy to Azure and can be reached with HTTP.
To compile and deploy your project to AWS:
boot aws/deploy -n <STACK_NAME>
Command checks that STACK_NAME
name is free. If it is free project
is deployed into Cloudformation stack with given name. If name
is reserved error message is shown. After build and deployment steps command print API endpoint base URL.
Visible HTTP endpoints of the project are in base URL.
Technical note: command check if S3 bucket with name
hedge-<STACK_NAME>
is free. This might be changed
in the future.
# Get information about the Azure Publishing Profile
boot azure/show-publish-profile -a functionapp -r resourcegroup
# Deploy to Azure and Persist the compiled artifacts in **target/** directory (index.js and function.json)
boot azure/deploy -a functionapp -r resourcegroup target
# Persist the compiled output. Given no options, defaults to Optimizations=simple and directory=target
boot azure/deploy-to-directory -O <optimization level> -f <function name> -d <directory>
# Deploy compiled artifacts from target directory (index.js and function.json)
boot azure/deploy-from-directory -a functionapp -r resourcegroup -d <directory>
# Get more help of task, i.e. commandline options
boot <task-name> -h