wash
helps you deal with all your remote or cloud-native infrastructure using the UNIX-y patterns and tools you already know and love!
Exploring, understanding, and inspecting modern infrastructure should be simple and straightforward. Whether it's containers, VMs, network devices, IoT stuff, or anything in between...they all have different ways of enumerating what you have, getting a stream of output, running commands, etc. Every vendor has its own tools and APIs that expose these features, each one different, each one bespoke. Thus, they are difficult to compose together to solve higher-level problems. And that's no fun at all!
UNIX's philosophy and abstractions have worked for decades. They're pretty good, and more importantly, they're familiar to millions of people. wash
intends to apply those same philosophies and abstractions to modern, distributed infrastructure: With wash
, we want to:
- make navigating stuff like servers, containers, or APIs as easy as navigating a local filesystem
- make scripting across your new-fangled infrastructure as easy as writing a local shell script
- render into text that which can be rendered into text (cuz text is a universal interface!) for easy viewing, editing, and UNIXy slicing-and-dicing
- build new versions of basic, UNIX tools to support the above goals (but reuse existing ones if they work!)
We're actively soliciting community feedback and input on our roadmap! Don't hesitate to file issues for new features, new plugin types, new primitives, new command-line tools, or anything else that crosses your mind. You can also chat with us directly on #wash
on Slack.
See the roadmap below to see what we've got planned!
We've implemented some neat features inside of wash
to support the above goals:
-
The
wash
daemon- presents a FUSE filesystem hierarchy for all of your resources, letting you navigate them in normal, filesystem-y ways
- preserves history of all executed commands, facilitating debugging
- serves up an HTTP API for everything
- caches information, for better performance
-
Primitives - the basic building blocks that form the foundation of
wash
, and dictate what kinds of things you can do to all the resourceswash
knows aboutlist
- lets you ask any resource what's contained inside of it, and what primitives it supports.- e.g. listing a Kubernetes pod returns its constituent containers
read
- lets you read the contents of a given resource- e.g. represent an EC2 instance's console output as a regular file you can open in a regular editor
stream
- gives you streaming-read access to a resource- e.g. to let you follow a container's output as its running
exec
- lets you execute a command against a resource- e.g. run a shell command inside a container, or on an EC2 vm, or on a routerOS device, etc.
-
CLI tools
wash ls
- a version ofls
that uses our API to enhance directory listings withwash
-specific info- e.g. show you what primitives are supported for each resource
wash meta
- emits a resource's metadata to standard outwash exec
- uses theexec
primitive to let you invoke commands against resourceswash find
- find resources using powerful selection predicates (WIP)wash tail -f
- follow updates to resources that support thestream
primitive as well as normal fileswash ps
- lists running processes on indicated compute instances that support theexec
primitivewash history
- lists all activity throughwash
;wash history <id>
can be used to view logs for a specific activitywash clear
- clears cached data for a subhierarchy rooted at the supplied path sowash
will re-request it
-
Core plugins (see the Roadmap below for more details)
docker
- presents a filesystem hierarchy of containers and volumes- found from the local socket or via
DOCKER
environment variables
- found from the local socket or via
kubernetes
- presents a filesystem hierarchy of pods, containers, and persistent volume claims- uses contexts from
~/.kube/config
- uses contexts from
aws
- presents a filesystem hierarchy for EC2 and S3- uses
AWS_SHARED_CREDENTIALS_FILE
environment variable or$HOME/.aws/credentials
andAWS_CONFIG_FILE
environment variable or$HOME/.aws/config
to find profiles and configure the SDK - IAM roles are supported when configured as described here. Note that currently
region
will also need to be specified with the profile. - if using MFA,
wash
will prompt for it on standard input. Credentials are valid for 1 hour. They are cached underwash/aws-credentials
in your user cache directory so they can be re-used across server restarts.wash
may have to re-prompt for a new MFA token in response to navigating thewash
environment to authorize a new session. - SSH is used to connect to EC2 instances. It relies on an
ssh-agent
for credentials, your SSH config to override the default user (ec2-user
) and port (22
), and~/.ssh/known_hosts
. If you don't want to add the target to yourknown_hosts
file, you can disable strict host key checking in your SSH config withHost *.compute.amazonaws.com StrictHostKeyChecking no
- uses
-
wash
allows for easy creation of out-of-process plugins using any language you want, frombash
togo
or anything in-between!wash
handles the plugin lifecycle. it invokes your plugin with a certain calling convention; all you have to do is supply the business logic- users interact with external plugins the exact same way as core plugins; they are first-class citizens
TBD: See GitHub releases.
Clone repo and within it run go install
.
Ensure $GOPATH/bin
is part of $PATH
.
Requires golang 1.12+.
If using iTerm2, we recommend installing iTerm2's shell integration to avoid issue#84.
Obtain FUSE for OSX here.
Add your mount directory to Spotlight's list of excluded directories to avoid heavy load.
Mount wash
's filesystem and API server with
wash server mnt
In another shell, navigate to mnt
to view available resources.
See available subcommands - such as ls
and exec
- with
wash help
When done, cd
out of mnt
, then run umount mnt
or Ctrl-C
the server process.
To get a sense of how wash
works, we've included a multi-node Docker application based on the Docker Compose tutorial. To start it run
docker-compose -f examples/swarm/docker-compose.yml up -d
When done, run
docker-compose -f examples/swarm/docker-compose.yml down
to stop the example application.
This starts a small Flask webapp that keeps a count of how often it's been accessed in a Redis instance that maintains state in a Docker volume.
Navigate the filesystem to view running containers
$ cd mnt/docker/containers
$ wash ls
NAME CREATED ACTIONS
./ <unknown> list
45a0265546d63a8f1b0d17033748db1468dc49dfd09cdaf2db62c45a60e82aaf/ 20 Mar 19 17:02 PDT exec, list, metadata
382776912d9373e6c4dc1350894b5290b22c36893a8fed08e2ba53fbb680c8a6/ 20 Mar 19 17:02 PDT exec, list, metadata
$ wash ls 382776912d9373e6c4dc1350894b5290b22c36893a8fed08e2ba53fbb680c8a6
NAME CREATED ACTIONS
./ 20 Mar 19 17:02 PDT exec, list, metadata
metadata.json <unknown> read
log <unknown> read, stream
Those containers are displayed as a directory, and provide access to their logs and metadata as files. Recent output from both can be accessed with common tools.
$ tail */log
==> 382776912d9373e6c4dc1350894b5290b22c36893a8fed08e2ba53fbb680c8a6/log <==
* Serving Flask app "app" (lazy loading)
* Environment: production
WARNING: Do not use the development server in a production environment.
Use a production WSGI server instead.
...
==> 45a0265546d63a8f1b0d17033748db1468dc49dfd09cdaf2db62c45a60e82aaf/log <==
1:C 21 Mar 2019 00:02:33.112 # oO0OoO0OoO0Oo Redis is starting oO0OoO0OoO0Oo
1:C 21 Mar 2019 00:02:33.112 # Redis version=5.0.4, bits=64, commit=00000000, modified=0, pid=1, just started
1:C 21 Mar 2019 00:02:33.112 # Configuration loaded
1:M 21 Mar 2019 00:02:33.113 * Running mode=standalone, port=6379.
...
The list earlier also noted that the container "directories" support the metadata action. We can get structured metadata in ether YAML or JSON with wash meta
$ wash meta 382776912d9373e6c4dc1350894b5290b22c36893a8fed08e2ba53fbb680c8a6 -o yaml
AppArmorProfile: ""
Args:
- app.py
Config:
...
We can interogate the container more closely with wash exec
$ wash exec 45a0265546d63a8f1b0d17033748db1468dc49dfd09cdaf2db62c45a60e82aaf whoami
root
Try exploring mnt/docker/volumes
to interact with the volume created for Redis.
All operations have their activity recorded to journals in wash/activity
under your user cache directory, identified by process ID and executable name.
wash
uses a user-specific cache directory to store running state. The user cache directory is $XDG_CACHE_HOME
or $HOME/.cache
on Unix systems, $HOME/Library/Caches
on macOS, and %LocalAppData%
on Windows.
If the wash
daemon exits with a exit status of 255, that typically means that wash
couldn't load the FUSE extensions. MacOS only allows for a certain (small) number of virtual devices on the system, and if all available slots are taken up by other programs then we won't be able to run. You can view loaded extensions with kextstat
. More information in this github issue for FUSE for macOS.
Project maintainers are not actively working on all of these things, but any of these are directions we would support others in pursuing.
- file/directory upload (prereq for executing commands that aren't just one-liners)
- edit a resource (e.g. edit a file representing a k8s ConfigMap, and upon write save it via the k8s api)
- delete a resource (e.g.
rm
-ing a file in an S3 bucket deletes it) - signal handling to represent basic verbs (e.g. sending a TERM to an EC2 instance will terminate it)
- copy / move / rename (how should this work?)
- make
stream
able to "go back in time" (e.g. supporttail -100 -f
style of "look-back")
- rad startup ASCII art logo (<- high priority!)
- expose plugin configuration via main config file
- expose what API calls are in-flight (to report status on large, distributed calls)
- colorized output for
ls
, similar toexa -l
- make
ls
emit something useful when used against non-wash
resources -
tail
that works forwash
resources that supportstream
-
exec
should work in parallel across multiple target resources -
find
that lets you refer towash
primitives (e.g. find all the resources under/docker
that supportexec
) - build an interactive shell that works over
exec
(need to update plugins API to support this, most likely) - a version of
top
that works usingwash
primitives to get information to display from multiple targets
list |
read |
stream |
exec |
meta |
|
---|---|---|---|---|---|
Docker | |||||
Containers | ✓ | ✓ | ✓ | ||
Container logs | ✓ | ✓ | |||
Volumes | ✓ | ✓ | ○ | ✓ | |
Images | ○ | ○ | |||
Networks | ○ | ○ | |||
Services | ○ | ○ | ○ | ○ | |
Stacks | ○ | ○ | |||
Swarm nodes | ○ | ○ | |||
Swarm config | ○ | ○ | ○ | ||
Kubernetes | |||||
Pods | ✓ | ✓ | ✓ | ✓ | ✓ |
Persistent Volume Claims | ✓ | ✓ | ✓ | ✓ | |
Services | ○ | ○ | |||
ConfigMaps | ○ | ○ | ○ | ||
generic k8s resources | ○ | ○ | |||
AWS | |||||
EC2 | ✓ | ✓ | ○ | ✓ | ✓ |
S3 buckets | ✓ | ✓ | |||
S3 directories | ✓ | ||||
S3 objects | ✓ | ✓ | ✓ | ||
Cloudwatch | ○ | ○ | ○ | ○ | |
Lambda | ○ | ○ | ○ | ○ | ○ |
pubsub (e.g. SNS) | ○ | ○ | ○ | ||
databases (e.g. dynamo, RDS) | ○ | ○ | ○ | ○ | ○ |
networking (e.g. ELB, Route53) | ○ | ○ | ○ | ○ | ○ |
SSH/WinRM targets | ○ | ○ | |||
SSHfs | ○ | ○ | ○ | ||
GCP | ○ | ○ | ○ | ○ | ○ |
Azure | ○ | ○ | ○ | ○ | ○ |
VMware | ○ | ○ | ○ | ○ | ○ |
Splunk | ○ | ○ | ○ | ||
Logstash | ○ | ○ | ○ | ||
Network Devices (e.g. Cisco) | ○ | ○ | ○ | ○ | ○ |
IoT (e.g. Nest, Hue, Rachio) | ○ | ○ | ○ | ○ | ○ |
wash itself (expose internals) |
○ | ○ | ○ | ○ | ○ |
✓ = Implemented ○ = Possible, but not yet implemented
We'd love to get contributions from you! For a quick guide, take a look at our guide to contributing.