Captain is yet another service discovery implementation based on redis. Captain sacrifices a little high availability for simplicity and performance. In most cases, we don't have tens of thousands of machines, the possibility of machine crashing is very low, high availability is not so abviously important. But the market only provides zookeeper/etcd/consul, they are complex, at least much complexer compared with captain. 中文文档
- Captain server is stateless, Client can connect to multiple captain servers.
- Captain client can be service provider and service consumer at the same time.
- If all captain server or redis shutdown, captain client will keep services information in local memory.
- If just one captain server shutdown, captain client will sync service information from other captain server.
- If redis master shutdown, you can alway dynamically switch to redis slave using web ui.
- Carefully monitor captain server and redis, recovery quickly, high availability still can be guaranteed.
- Service List is saved into redis as sortedset with key equals host:port and score equals ${now + ttl}
- Expiring Thread will periodically trim the expired items of this sortedset
- To track the changes, redis keeps a global version and each sub version for every service list.
- Client will periodically check the global version, If the global version changed, client will check all version of dependent services, if any dependent service changes, client will reload the changed service list.
- Less Keys, Large Value. Though captain provided keyvalue api, KeyValue should not be overused. Captain server lacks the ability to handle too much keys. Use it only for global configuration and use it less. you can design a large Value instead.
- Client interact with captain server with http api only
- keep service /api/service/keep?name=sample&host=localhost&port=6000&ttl=30 GET
- cancel service /api/service/cancel?name=sample&host=localhost&port=6000 GET
- get service version /api/service/version?name=sample1&name=sample2 GET
- get service list /api/service/set?name=sample GET
- set /api/kv/set?key=sample&value={"a": "m", "b": "n", "c": {"a": "m", "b": "n"}} POST
- get /api/kv/get?key=sample GET
- mget /api/kv/mget?key=sample1&key=sample2 GET
- get kv version /api/kv/version?key=sample1&key=sample2 GET
- get global service & kv version in single api /api/version
install redis
install java8
install maven
git clone github.com/pyloque/captain.git
cd captain
mvn -DskipTests package
java -jar target/captain.jar
java -jar target/captain.jar ${configfile} # custom config file
open web ui
http://localhost:6789/ui/
Default Config File is ${user.home}/.captain/captain.ini
[server]
host = 0.0.0.0
port = 6789
thread = 24 # sparkjava threadpool size
[redis]
host = localhost
port = 6379
db = 0
[watch]
interval = 1000 # service expiring check interval, default 1000ms. server will run in readonly mode if interval=0.
- service expiring check thread will not be started.
- service keep and cancel api will not be open
When redis master oncasionally crashed, you should immediately switch to redis slave. After switched, server will enter in urgent mode.Expring Watcher Thread is closed to keep services list always alive. After all captain servers are switched, click "start watcher" in web ui, to make the server quit urgent mode.
- Python Client https://github.com/pyloque/pycaptain
- Java Client https://github.com/pyloque/captain-java
- Golang Client https://github.com/pyloque/gocaptain
- Agent Server https://github.com/pyloque/captain-agent
- PHP Client https://github.com/pyloque/phpcaptain
- Wonderful SparkJava https://github.com/perwendel/spark/
- Simple & Direct ini4j http://ini4j.sourceforge.net/
Lack of long running experience, Captain's stability has not been proved yet. But captain's design is very simple and clear, if you do not want to try captain with bugs, just make one yourself.
<img src="screenshot/weixin.png" width="320" title="QR"/ >