Ever say to yourself, "Darn, I wish I had a simple HTTP app to test my infrastructure—specifically out-of-app service discovery?" Well, testapp
is your answer! With its patent-pending dependency-chain-calling technology, you too can mimic a deep service call.
testapp
is a simple Flask application designed to test web apps calling other web apps. It has three modes: stand-alone mode, static-dependency mode, and dynamic-dependency mode.
For testing:
git clone https://github.com/thoughtarray/testapp.git
cd testapp
virtualenv venv
# or python[2,2.7] -m virtualenv venv
. venv/bin/activate
python testapp --name foo --port 5000
Not very useful. It just returns what you tell it to.
testapp --name foo --port 5000
When launching testapp
in this mode, dependent apps must be hard-configured via arguments. It might look something like this:
testapp --name foo --port 5000 \
--static "bar=http://localhost:5001/"
--static "baz=http://localhost:5002/"
When launching testapp
in this mode, dependent apps don't have to be known. It might look something like this:
testapp --name foo --port 5000 --dynamic "http://localhost:80/{}/"
# --- or ---
testapp --name foo --port 5000 --dynamic "http://{}.foo.com:80/"
# --- or ---
testapp --name foo --port 5000 --dynamic "http://localhost:80/" \
--header "X-Route={}"
This mode would have to be supported by some kind of proxy.
testapp [-nprsd]
-n, --name string Name of instance; default: testapp
--host IP host to accept requests; default: 127.0.0.1
-p, --port PORT Port of instance; default: 80
-r, --return CODE:STRING Success return; default: 200:NameOfInstance
-s, --static NAME=URL Starts static-dependency mode (additive property)
-d, --dynamic URL Starts dynamic-dependency mode; URL may contain "{}"
-h, --header KEY=VALUE HTTP header (additive property); may use "{}" in dynamic-dependency mode
The simplest use of testapp
would be a single instance in stand-alone mode. This use may be good for testing something such as Docker or Habitat.
testapp --name hello --port 5000 --return "200:Hello, world!"
curl -w "\n" "localhost:5000"
# Should return "Hello, world!"
A more complex use would be to form a static chain:
# Instance 1
testapp --name foo --port 5000 \
--static "bar=http://localhost:5001/"
# Instance 2
testapp --name bar --port 5001
curl -w "\n" "localhost:5000"
# Should return "bar"
When an instance has a single, static dependency, the default action when called is to call its dependency as opposed to return its name (or --return data).
"bar" was returned because it was the only dependency of the foo testapp
.
More complex relational structures such as trees or graphs require a special type of request: a chain-script request. I know this sound fancy, but it isn't. Here is an example:
# Instance 1
testapp --name foo --port 5000 \
--static "bar=http://localhost:5001/"
--static "baz=http://localhost:5002/"
# Instance 2
testapp --name bar --port 5001 \
--static "foo=http://localhost:5000/"
--static "baz=http://localhost:5002/"
# Instance 3
testapp --name baz --port 5002 \
--static "bar=http://localhost:5001/"
--static "foo=http://localhost:5000/"
curl -w "\n" "localhost:5000?chain=bar,baz,foo"
# Should return ["foo", "bar", "baz", "foo"]
The query parameter "chain" is a comma-separated script of what each testapp
should call next. In the example, the following call chain results:
you -> foo -> bar -> baz -> foo (deepest dependency) -> baz -> bar -> foo -> you
As you can see, the return is a JSON array of each node's return value.
The purpose of this mode is to test out various external-to-app service discovery tools that use proxies or DNS.
Assume the use SmartStack or Consul Template along with Nginx or HAProxy and proxying via a header:
# Start a proxy; I use Docker here for convenience
docker run -d --name proxy \
--add-host docker-host:<ip of host> \
-p 5000:5000 \
-v `pwd`/example/nginx.conf:/etc/nginx/conf.d/default.conf:ro \
nginx
# Instance 1
testapp --name foo --host 0.0.0.0 --port 5050 \
--dynamic "http://localhost:5000/" \
--header "X-Route={}"
# Instance 2
testapp --name bar --host 0.0.0.0 --port 5051 \
--dynamic "http://localhost:5000/" \
--header "X-Route={}"
# Instance 3
testapp --name baz --host 0.0.0.0 --port 5052 \
--dynamic "http://localhost:5000/" \
--header "X-Route={}"
curl -w "\n" -H "X-Route: foo" "localhost:5000?chain=foo,bar,baz"
# Should return ["foo", "bar", "baz"]