Merge pull request #2 from phanikishoreg/cleanup
chore: clean up and comment build scripts and improve READMEmain
commit
81950fcf94
@ -1,80 +1,138 @@
|
||||
# aWsm (awesome)
|
||||
# aWsm (awesome)
|
||||
|
||||
This project is a work-in-progress to build an efficient WASM runtime, **aWsm**, using `silverfish` compiler.
|
||||
**aWsm** is an efficient serverless runtime built with the `silverfish` compiler. It combines WebAssembly sandboxing with asynchronous I/O to provide a lightweight serverless solution suitable for edge computing.
|
||||
|
||||
## Setting up the environment
|
||||
## Host Dependencies
|
||||
|
||||
- Docker - [Installation Instructions](https://docs.docker.com/install/)
|
||||
- libuv
|
||||
|
||||
If on Debian, you can install libuv with the following:
|
||||
|
||||
To use a Docker container based environment, that makes your life easy by installing all the required dependencies and builds the toolchain for you.
|
||||
Run
|
||||
```bash
|
||||
./devenv.sh install_libuv
|
||||
```
|
||||
|
||||
## Setting up the environment
|
||||
|
||||
**Note: These steps require Docker. Make sure you've got it installed!**
|
||||
|
||||
We provide a Docker build environment configured with the dependencies and toolchain needed to build the aWsm runtime and serverless functions.
|
||||
|
||||
To setup this environment, run:
|
||||
|
||||
```bash
|
||||
./devenv.sh setup
|
||||
```
|
||||
**make sure you've docker installed.**
|
||||
|
||||
To enter the docker environment,
|
||||
```
|
||||
## Using the Docker container to compile your serverless functions
|
||||
|
||||
To enter the docker environment, run:
|
||||
|
||||
```bash
|
||||
./devenv.sh run
|
||||
```
|
||||
**spawns a shell in the container.**
|
||||
|
||||
To setup toolchain path (within a container, per `run`)
|
||||
The first time you enter this environment, run the following to copy the awsmrt binary to /awsm/runtime/bin.
|
||||
|
||||
```bash
|
||||
cd /awsm/runtime
|
||||
make clean all
|
||||
```
|
||||
source /opt/awsm/bin/devenv_src.sh
|
||||
|
||||
There are a set of benchmarking applications in the `/awsm/runtime/tests` directory. Run the following to compile all benchmarks runtime tests using silverfish and then copy all resulting `<application>_wasm.so` files to /awsm/runtime/bin.
|
||||
|
||||
```bash
|
||||
cd /awsm/runtime/tests/
|
||||
make clean all
|
||||
```
|
||||
|
||||
## To run applications
|
||||
You now have everything that you need to execute your first serverless function on aWsm
|
||||
|
||||
There are a set of benchmarking applications in `code_benches` directory that should be "loadable", WIP!!
|
||||
**All the remaining steps are in a Docker container environment.**
|
||||
To exit the container:
|
||||
|
||||
```bash
|
||||
exit
|
||||
```
|
||||
cd /awsm/tests/
|
||||
|
||||
make clean all
|
||||
```
|
||||
This compiles all benchmarks in silverfish and other runtime tests and copies `<application>_wasm.so` to /awsm/runtime/bin.
|
||||
To stop the Docker container:
|
||||
|
||||
```bash
|
||||
./devenv.sh stop
|
||||
```
|
||||
cd /awsm/runtime
|
||||
make clean all
|
||||
```
|
||||
This will copy the awsmrt binary to /awsm/runtime/bin.
|
||||
|
||||
## Running your first serverless function
|
||||
|
||||
An aWsm serverless function consists of a shared library (*.so) and a JSON configuration file that determines how the runtime should execute the serverless function. As an example, here is the configuration file for our sample fibonacci function:.
|
||||
|
||||
```json
|
||||
{
|
||||
"active": "yes",
|
||||
"name": "fibonacci",
|
||||
"path": "fibonacci_wasm.so",
|
||||
"port": 10000,
|
||||
"argsize": 1,
|
||||
"http-req-headers": [],
|
||||
"http-req-content-type": "text/plain",
|
||||
"http-req-size": 1024,
|
||||
"http-resp-headers": [],
|
||||
"http-resp-size": 1024,
|
||||
"http-resp-content-type": "text/plain"
|
||||
}
|
||||
```
|
||||
cd /awsm/runtime/bin
|
||||
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:`pwd`
|
||||
```
|
||||
|
||||
**Create input and test files**
|
||||
Supports module registration using json format now and invocations as well.
|
||||
More importantly, each module runs a udp server and waits for connections.
|
||||
The udpclient tool in `runtime/tools` directory uses a format `<ip:port>$<json_request_string>`,
|
||||
connects to <ip:port> and sends the <json_reqest_string> to the IP address it connects at the start.
|
||||
The `port` and `name` fields are used to determine the path where our serverless function will be served served.
|
||||
|
||||
In our case, we are running the aWsm runtime on localhost, so our function is available at `http://localhost:10000/fibonacci`
|
||||
|
||||
To run `awsm runtime`,
|
||||
Our fibonacci function will parse a single argument from the HTTP POST body that we send. The expected Content-Type is "text/plain" and the buffer is sized to 1024 bytes for both the request and response. This is sufficient for our simple Fibonacci function, but this must be changed and sized for other functions, such as image processing.
|
||||
|
||||
Now that we understand roughly how the aWsm runtime interacts with serverless function, let's run Fibonacci!
|
||||
|
||||
From the root project directory of the host environment (not the Docker container!), navigate to the binary directory
|
||||
|
||||
```bash
|
||||
cd runtime/bin/
|
||||
```
|
||||
./awsmrt ../tests/test_modules.json
|
||||
|
||||
Now run the awsmrt binary, passing the JSON file of the serverless function we want to serve. Because serverless functions are loaded by aWsm as shared libraries, we want to add the `runtime/tests/` directory to LD_LIBRARY_PATH.
|
||||
|
||||
```bash
|
||||
LD_LIBRARY_PATH="$(pwd):$LD_LIBRARY_PATH" ./awsmrt ../tests/test_fibonacci.json
|
||||
```
|
||||
|
||||
To run the udpclient,
|
||||
While you don't see any output to the console, the runtime is running in the foreground.
|
||||
|
||||
Let's now invoke our serverless function to compute the 10th fibonacci number. I'll use [HTTPie](https://httpie.org/) to send a POST request with a body containing the parameter I want to pass to my serverless function. Feel free to use cURL or whatever network client you prefer!
|
||||
|
||||
Open a new terminal session and execute the following
|
||||
|
||||
```bash
|
||||
echo "10" | http :10000
|
||||
```
|
||||
./udpclient ../tests/test_sandboxes.jsondata
|
||||
|
||||
You should receive the following in response. The serverless function says that the 10th fibonacci number is 55, which seems to be correct!
|
||||
|
||||
```bash
|
||||
HTTP/1.1 200 OK
|
||||
Content-length: 3
|
||||
Content-type: text/plain
|
||||
|
||||
55
|
||||
```
|
||||
And follow the prompts in udpclient to send requests to the runtime.
|
||||
|
||||
## WIP (Work In Progress)
|
||||
When done, terminal the aWsm runtime with `Ctrl+c`
|
||||
|
||||
## Removing the aWsm Runtime
|
||||
|
||||
If you are finished working with the aWsm runtime and wish to remove it, run the following command to delete our Docker build and runtime images.
|
||||
|
||||
```bash
|
||||
./devenv.sh rma
|
||||
```
|
||||
|
||||
* ~~Dynamic loading of multiple modules~~
|
||||
* ~~Multiple sandboxes (includes multiple modules, multiple instances of a module)~~
|
||||
* ~~Bookkeeping of multiple modules and multiple sandboxes.~~
|
||||
* ~~Runtime to "poll"?? on requests to instantiate a module~~ and respond with the result.
|
||||
* ~~Runtime to schedule multiple sandboxes.~~
|
||||
* Efficient scheduling and performance optimizations.
|
||||
* ~~Runtime to enable event-based I/O (using `libuv`).~~ (basic I/O works with libuv)
|
||||
* To enable WASI interface, perhaps through the use of WASI-SDK
|
||||
And then simply delete this repository.
|
||||
|
||||
## Silverfish compiler
|
||||
## Problems or Feedback?
|
||||
|
||||
Silverfish compiler uses `llvm` and interposes on loads/stores to enable sandbox isolation necessary in `aWsm` multi-sandboxing runtime.
|
||||
`aWsm` runtime includes the compiler-runtime API required for bounds checking in sandboxes.
|
||||
Most of the sandboxing isolation is copied from the silverfish runtime.
|
||||
If you encountered bugs or have feedback, please let us know in our [issue tracker.](https://github.com/phanikishoreg/awsm-Serverless-Framework/issues)
|
||||
|
@ -1,144 +1,198 @@
|
||||
#!/bin/sh
|
||||
|
||||
#
|
||||
# This environment file and dockerfile are
|
||||
# inspired by lucet's devenv_xxx.sh scripts
|
||||
#
|
||||
HOST_ROOT=${HOST_ROOT:-$(cd "$(dirname ${BASH_SOURCE:-$0})" && pwd)}
|
||||
# This environment file and dockerfile are inspired by lucet's devenv_xxx.sh scripts
|
||||
|
||||
# Root directory of host
|
||||
HOST_ROOT=${HOST_ROOT:-$(cd "$(dirname "${BASH_SOURCE:-$0}")" && pwd)}
|
||||
|
||||
# Name use to represent the Awsm system
|
||||
SYS_NAME='awsm'
|
||||
|
||||
# /awsm
|
||||
HOST_SYS_MOUNT=${HOST_SYS_MOUNT:-"/${SYS_NAME}"}
|
||||
|
||||
SYS_WASMCEPTION='silverfish/wasmception'
|
||||
|
||||
# awsm
|
||||
SYS_DOC_NAME=${SYS_NAME}
|
||||
|
||||
# awsm-dev
|
||||
SYS_DOC_DEVNAME=${SYS_DOC_NAME}'-dev'
|
||||
|
||||
# Docker Tag we want to use
|
||||
SYS_DOC_TAG='latest'
|
||||
|
||||
# The name of the non-dev Docker container that we want to build. awsm:latest
|
||||
SYS_DOC_NAMETAG=${SYS_DOC_NAME}:${SYS_DOC_TAG}
|
||||
SYS_DOC_DEVNAMETAG=${SYS_DOC_DEVNAME}:${SYS_DOC_TAG}
|
||||
SYS_BUILD_TIMEOUT=10
|
||||
|
||||
usage()
|
||||
{
|
||||
echo "usage $0 <setup/run/stop>"
|
||||
# An optional timeout that allows a user to terminate the script if awsm-dev is detected
|
||||
SYS_BUILD_TIMEOUT=0
|
||||
|
||||
# Provides help to user on how to use this script
|
||||
usage() {
|
||||
echo "usage $0 <setup||run||stop||rm||rma/>"
|
||||
echo " setup Build aWsm and a Docker container with toolchain needed to compile your own functions"
|
||||
echo " run Start the aWsm Docker image as an interactive container with this repository mounted"
|
||||
echo " stop Stop and remove the aWsm Docker container after use"
|
||||
echo " rm Remove the aWsm runtime container and image, but leaves the awsm-dev container in place"
|
||||
echo " rma Removes all the awsm and awsm-dev containers and images"
|
||||
}
|
||||
|
||||
countdown()
|
||||
{
|
||||
tmp_cnt=$1
|
||||
while [ ${tmp_cnt} -gt 0 ]; do
|
||||
printf "${tmp_cnt}."
|
||||
# sleep 1
|
||||
tmp_cnt=$((tmp_cnt - 1))
|
||||
done
|
||||
echo
|
||||
# Given a number of seconds, initiates a countdown sequence
|
||||
countdown() {
|
||||
tmp_cnt=$1
|
||||
while [ "${tmp_cnt}" -gt 0 ]; do
|
||||
printf "%d." "${tmp_cnt}"
|
||||
sleep 1
|
||||
tmp_cnt=$((tmp_cnt - 1))
|
||||
done
|
||||
echo
|
||||
}
|
||||
|
||||
envsetup()
|
||||
{
|
||||
if docker image inspect ${SYS_DOC_NAMETAG} > /dev/null; then
|
||||
echo "${SYS_DOC_NAMETAG} image exists, remove it first!"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
echo "Setting up ${SYS_NAME}"
|
||||
git submodule update --init --recursive 2>/dev/null ||:
|
||||
|
||||
# Perhaps change in the wasmception (forked) repo,
|
||||
# Gregor already forked every mainline repo to modify something or the other!
|
||||
#
|
||||
# That said, you may comment this if you're not behind a firewall!
|
||||
# http:// doesn't work for me at my current work place.
|
||||
echo "Changing http:// to https:// in ${SYS_WASMCEPTION}"
|
||||
sed -i 's/http:\/\//https:\/\//' ${SYS_WASMCEPTION}/Makefile
|
||||
|
||||
if docker image inspect ${SYS_DOC_DEVNAMETAG} >> /dev/null; then
|
||||
echo "${SYS_DOC_DEVNAME} image exists, rebuilding it"
|
||||
echo "(you have ${SYS_BUILD_TIMEOUT}secs to stop the rebuild)"
|
||||
countdown ${SYS_BUILD_TIMEOUT}
|
||||
fi
|
||||
|
||||
echo "Building ${SYS_DOC_DEVNAMETAG}"
|
||||
docker build -t ${SYS_DOC_DEVNAMETAG} .
|
||||
|
||||
echo "Creating ${SYS_DOC_NAMETAG} on top of ${SYS_DOC_DEVNAMETAG}"
|
||||
docker run --privileged --name=${SYS_DOC_DEVNAME} --detach --mount type=bind,src="$(cd $(dirname ${0}); pwd -P),target=/${SYS_NAME}" \
|
||||
${SYS_DOC_DEVNAMETAG} /bin/sleep 99999999 > /dev/null
|
||||
|
||||
echo "Building ${SYS_NAME}"
|
||||
docker exec -t -w ${HOST_SYS_MOUNT} ${SYS_DOC_DEVNAME} make install
|
||||
|
||||
echo "Tagging the new image"
|
||||
docker container commit ${SYS_DOC_DEVNAME} ${SYS_DOC_NAMETAG}
|
||||
|
||||
echo "Cleaning up ${SYS_DOC_DEVNAME}"
|
||||
docker kill ${SYS_DOC_DEVNAME}
|
||||
docker rm ${SYS_DOC_DEVNAME}
|
||||
|
||||
echo "Done!"
|
||||
# Build and runs the build container awsm-dev and then executes make install on the project
|
||||
# Finally "forks" the awsm-dev build container into the awsm execution container
|
||||
envsetup() {
|
||||
# I want to create this container before the Makefile executes so that my user owns it
|
||||
# This allows me to execute the awsmrt binary from my local host
|
||||
mkdir -p "$HOST_ROOT/runtime/bin"
|
||||
|
||||
# Check to see if the awsm:latest image exists, exiting if it does
|
||||
# Because awsm:latest is "forked" after completing envsetup, this suggests that envsetup was already run
|
||||
if docker image inspect ${SYS_DOC_NAMETAG} 1>/dev/null 2>/dev/null; then
|
||||
echo "${SYS_DOC_NAMETAG} image exists, which means that 'devenv.sh setup' already ran to completion!"
|
||||
echo "If you are explicitly trying to rebuild Awsm, run the following:"
|
||||
echo "devenv.sh rma | Removes the images awsm:latest AND awsm-dev:latest"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
echo "Setting up ${SYS_NAME}"
|
||||
|
||||
echo "Updating git submodules"
|
||||
git submodule update --init --recursive 2>/dev/null || :d
|
||||
|
||||
# Downstream fix to force use of https in the Wasmception Makefile
|
||||
# TODO: Should this be moved upstream?
|
||||
echo "Patching ${SYS_WASMCEPTION}/Makefile to use https:// in place of http://"
|
||||
sed -i 's/http:\/\//https:\/\//' ${SYS_WASMCEPTION}/Makefile
|
||||
|
||||
# As a user nicety, warn the user if awsm-dev is detected
|
||||
# This UX differs from detecting awsm, which immediately exits
|
||||
# This is disabled because it doesn't seem useful
|
||||
if
|
||||
docker image inspect "${SYS_DOC_DEVNAMETAG}" 1>/dev/null 2>/dev/null && [ $SYS_BUILD_TIMEOUT -gt 0 ]
|
||||
then
|
||||
echo "${SYS_DOC_DEVNAME} image exists, rebuilding it"
|
||||
echo "(you have ${SYS_BUILD_TIMEOUT}secs to stop the rebuild)"
|
||||
countdown ${SYS_BUILD_TIMEOUT}
|
||||
fi
|
||||
|
||||
# Build the image awsm-dev:latest
|
||||
echo "Building ${SYS_DOC_DEVNAMETAG}"
|
||||
docker build --tag "${SYS_DOC_DEVNAMETAG}" .
|
||||
|
||||
# Run the awsm-dev:latest image as a background container named awsm-dev with the project directly mounted at /awsm
|
||||
echo "Creating the build container ${SYS_DOC_NAMETAG} from the image ${SYS_DOC_DEVNAMETAG}"
|
||||
docker run \
|
||||
--privileged \
|
||||
--name=${SYS_DOC_DEVNAME} \
|
||||
--detach \
|
||||
--mount type=bind,src="$(cd "$(dirname "${0}")" && pwd -P || exit 1),target=/${SYS_NAME}" \
|
||||
"${SYS_DOC_DEVNAMETAG}" /bin/sleep 99999999 >/dev/null
|
||||
|
||||
# Execute the make install command on the awsm-dev image to build the project
|
||||
echo "Building ${SYS_NAME}"
|
||||
docker exec \
|
||||
--tty \
|
||||
--workdir "${HOST_SYS_MOUNT}" \
|
||||
${SYS_DOC_DEVNAME} make install
|
||||
|
||||
# Create the image awsm:latest from the current state of docker-dev
|
||||
echo "Tagging the new image"
|
||||
docker container commit ${SYS_DOC_DEVNAME} ${SYS_DOC_NAMETAG}
|
||||
|
||||
# Kill and remove the running awsm-dev container
|
||||
echo "Cleaning up ${SYS_DOC_DEVNAME}"
|
||||
docker kill ${SYS_DOC_DEVNAME}
|
||||
docker rm ${SYS_DOC_DEVNAME}
|
||||
|
||||
echo "Done!"
|
||||
}
|
||||
|
||||
envrun()
|
||||
{
|
||||
if ! docker image inspect ${SYS_DOC_NAMETAG} > /dev/null; then
|
||||
envsetup
|
||||
fi
|
||||
|
||||
if docker ps -f name=${SYS_DOC_NAME} --format '{{.Names}}' | grep -q "^${SYS_DOC_NAME}" ; then
|
||||
echo "Container is running" >&2
|
||||
else
|
||||
echo "Starting ${SYS_DOC_NAME}"
|
||||
docker run --privileged --security-opt seccomp:unconfined --name=${SYS_DOC_NAME} --detach --mount type=bind,src="$(cd $(dirname ${0}); pwd -P),target=/${SYS_NAME}" \
|
||||
${SYS_DOC_NAMETAG} /bin/sleep 99999999 > /dev/null
|
||||
fi
|
||||
|
||||
echo "Running shell"
|
||||
docker exec -t -i -w "${HOST_SYS_MOUNT}" ${SYS_DOC_NAME} /bin/bash
|
||||
# Executes an interactive BASH shell in the awsm container with /awsm as the working directory
|
||||
# This is the Awsm project directory mounted from the host environment.
|
||||
# If the image awsm:latest does not exist, automatically runs envsetup to build awsm and create it
|
||||
# If the a container names awsm is not running, starts it from awsm:latest, mounting the Awsm project directory to /awsm
|
||||
envrun() {
|
||||
if ! docker image inspect ${SYS_DOC_NAMETAG} >/dev/null; then
|
||||
envsetup
|
||||
fi
|
||||
|
||||
if docker ps -f name=${SYS_DOC_NAME} --format '{{.Names}}' | grep -q "^${SYS_DOC_NAME}"; then
|
||||
echo "Container is running" >&2
|
||||
else
|
||||
|
||||
echo "Starting ${SYS_DOC_NAME}"
|
||||
docker run \
|
||||
--privileged \
|
||||
--security-opt seccomp:unconfined \
|
||||
--name=${SYS_DOC_NAME} \
|
||||
--detach \
|
||||
--mount type=bind,src="$(cd "$(dirname "${0}")" && pwd -P || exit 1),target=/${SYS_NAME}" \
|
||||
${SYS_DOC_NAMETAG} /bin/sleep 99999999 >/dev/null
|
||||
fi
|
||||
|
||||
echo "Running shell"
|
||||
docker exec --tty --interactive --workdir "${HOST_SYS_MOUNT}" ${SYS_DOC_NAME} /bin/bash
|
||||
}
|
||||
|
||||
envstop()
|
||||
{
|
||||
echo "Stopping container"
|
||||
docker stop ${SYS_DOC_NAME}
|
||||
echo "Removing container"
|
||||
docker rm ${SYS_DOC_NAME}
|
||||
# Stops and removes the awsm "runtime" container
|
||||
envstop() {
|
||||
echo "Stopping container"
|
||||
docker stop ${SYS_DOC_NAME}
|
||||
echo "Removing container"
|
||||
docker rm ${SYS_DOC_NAME}
|
||||
}
|
||||
|
||||
envrm()
|
||||
{
|
||||
envstop
|
||||
docker rmi ${SYS_DOC_NAME}
|
||||
# Stops and removes the awsm "runtime" container and then removes the awsm "runtime" image
|
||||
envrm() {
|
||||
envstop
|
||||
docker rmi ${SYS_DOC_NAME}
|
||||
}
|
||||
|
||||
envrma()
|
||||
{
|
||||
envrm
|
||||
docker rmi ${SYS_DOC_DEVNAME}
|
||||
# Stops and removes the awsm "runtime" container and image and then removes the awsm-dev "build image" image
|
||||
envrma() {
|
||||
envrm
|
||||
docker rmi ${SYS_DOC_DEVNAME}
|
||||
}
|
||||
|
||||
if [ $# -ne 1 ]; then
|
||||
usage $0
|
||||
exit 1
|
||||
echo "incorrect number of arguments: $*"
|
||||
usage "$0"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
case $1 in
|
||||
run)
|
||||
envrun
|
||||
;;
|
||||
stop)
|
||||
envstop
|
||||
;;
|
||||
setup)
|
||||
envsetup
|
||||
;;
|
||||
rm)
|
||||
envrm
|
||||
;;
|
||||
rma)
|
||||
envrma
|
||||
;;
|
||||
*)
|
||||
echo "invalid option: $1"
|
||||
usage $0
|
||||
exit 1
|
||||
;;
|
||||
run)
|
||||
envrun
|
||||
;;
|
||||
stop)
|
||||
envstop
|
||||
;;
|
||||
setup)
|
||||
envsetup
|
||||
;;
|
||||
rm)
|
||||
envrm
|
||||
;;
|
||||
rma)
|
||||
envrma
|
||||
;;
|
||||
*)
|
||||
echo "invalid option: $1"
|
||||
usage "$0"
|
||||
exit 1
|
||||
;;
|
||||
esac
|
||||
echo
|
||||
echo "done!"
|
||||
|
Loading…
Reference in new issue