ayd

command module
v0.14.0 Latest Latest
Warning

This package is not in the latest version of its module.

Go to latest
Published: May 15, 2022 License: MIT Imports: 24 Imported by: 0

README

CI test status Code Climate maintainability Codecov Test Coverage Docker Build Status

The easiest status monitoring service to check if your service is dead or alive.

Features

Good at

  • Make a status page for temporary usage.

    You can start it via one command! And, stop it via just Ctrl-C!

  • Make a status page for minimal systems.

    Single binary server, single log file, there is no database!

Not good at

  • Complex customization, extension.

    There is a few extension way, but extensibility is not the goal of this project.

  • Investigate more detail.

    This program is just for check your service is dead or alive.

Quick start

  1. Download the latest version from release page.

  2. Extract downloaded package and put to somewhere that registered to PATH.

  3. Run the server with target URLs (and schedule if need) as arguments.

$ ayd https://your-service.example.com ping:another-host.example.com
  1. Check your status page.

Usage detail

Status page and endpoints

Ayd has these pages/endpoints.

path description
/status.html Human friendly status page in HTML.
/status.txt Human friendly status page in plain text.
/status.json Machine readable status page in JSON format.
/incidents.html Human friendly incident history page in HTML.
/incidents.rss Incident history feed in RSS 2.0 format.
/incidents.csv Incident history in CSV format.
/incidents.json Incident history in JSON format.
/log.html Raw log data in HTML page.
/log.tsv Raw log file in TSV format.
/log.csv Raw log file in CSV format.
/log.json Raw log file in JSON format.
/targets.txt The list of target URLs, separated by \n.
/targets.json The list of target URLs in JSON format.
/metrics Minimal status page for use by Prometheus.
/healthz Health status page for checking status of Ayd itself.
Change encoding of /stauts.txt

/status.txt accepts charset query to changing the encoding of result. The query value is unicode or ascii. In default, the encoding is unicode.

examples:

Be careful, the target URL or the message won't convert even if set charset=ascii. The response could include non-ascii text.

Filter log entries in /log.html, /log.tsv, /log.csv, and /log.json

The log endpoints accept since, until, target, and query query to filtering log entries.

since and until is the queries to filtering by date-time, in RFC3339 format like 2001-02-03T16:05:06+09:00. In default, Ayd replies logs that from 7 days ago to current time.

And, target is the query to filtering by target URL.

query is space delimited query for filtering records. It works as perfect matching for status, partial match for target URL and message text. And, you can use filters for latency like <10ms or >=1s.

examples:

Supported schemes

The target services or the alerting settings are specified in the arguments of ayd command as a URL. Ayd supports below schemes.

for Target for Alert
http: / https:
ftp: / ftps:
ping:
tcp:
dns:
exec:
source:

You can use extra scheme with plugin if you want to.

http: / https:

Fetch HTTP/HTTPS page and check if the status code is 2xx or not.

You can use GET, HEAD, POST, OPTIONS, and CONNECT method by specifying like http-post://... or https-head://.... The default method is GET.

Ayd will Follow redirect maximum 10 times.

HTTP will timeout in 10 minutes and report as failure.

examples:

  • http://example.com/
  • https://example.com/
  • http-head://example.com/path/to/somewhere
  • https-options://example.com/abc?def=ghi
for Alert

In alerting, Ayd adds some queries to send information about the incident.

query name example description
ayd_checked_at 2001-02-03T16:05:06+09:00 The timestamp when status changed
ayd_status FAILURE, DEGRADE, UNKNOWN, HEALTHY The current status of the target
ayd_latency 123.456 The latency of the latest checking
ayd_target https://target.example.com The target URL
ayd_message The latest message of the target
ftp: / ftps:

Send LIST or MLSD command of FTP and check the result.

It uses anonymous as username and password if absent those in the URL.

FTP will timeout in 10 minutes and report as failure.

examples:

  • ftp://example.com/
  • ftp://example.com/path/to/directory
  • ftps://foo:bar@example.com/path/to/file.txt
ping:

Send ICMP echo request (a.k.a. ping command) and check if the target is connected or not.

Ayd sends 3 packets in 1 second and expects all packets to return. These parameter can changed by AYD_PING_PACKETS and AYD_PING_PERIOD environment variable.

In Linux or MacOS, Ayd use non-privileged ICMP in default. So, you can use ping even if rootless. But this way is not work on some platforms for instance docker container. Please set yes to AYD_PRIVILEGED environment variable to use privileged ICMP.

You can specify IPv4 or IPv6 with ping4: or ping6: scheme.

Ping will timeout in 30 seconds after sent all packets and report as failure.

examples:

  • ping:example.com
  • ping:192.168.1.1
  • ping:192.168.1.10#my-server
tcp:

Connect to TCP and check if the service is listening or not.

tcp:// will select IPv4 or IPv6 automatically. But you can also use tcp4:// or tcp6:// to choose IP protocol version.

TCP will timeout in 10 seconds and report as failure.

examples:

  • tcp://example.com:3309
  • tcp4://127.0.0.1:3309
  • tcp6://[::1]:3309
  • tcp://192.168.1.10:80#my-server
dns:

Resolve hostname via DNS and check if the host exists or not.

You can specify record type as a type query like dns:example.com?type=A, or as a scheme like dns-cname:example.com. Supported type is A, AAAA, CNAME, MX, NS, and TXT.

There are dns4: and dns6: scheme for shorthand of A and AAAA type.

You also can specify the DNS server as a host name of URL like dns://8.8.8.8/example.com.

DNS will timeout in 10 seconds and report as failure.

examples:

  • dns:example.com
  • dns4:example.com
  • dns:example.com?type=AAAA
  • dns-cname:example.com
  • dns://8.8.8.8/example.com
exec:

Execute external command and check if the exit code is 0 or not. The exit code 0 means HEALTHY, otherwise mean FAILURE. If couldn't execute command, Ayd reports as UNKNOWN.

The command's stdout and stderr will be captured as a message of the status check record. You should keep output as short as possible because Ayd is not good at record a long message.

You can specify the first argument as the fragment of URL like below.

exec:/path/to/command#this-is-argument

Above target URL works the same as the below command in the shell.

$ /path/to/command this-is-argument

And, you can specify environment arguments as the query of URL like below.

exec:/path/to/command?something=foobar&hello=world

Above target URL works the same as the below command in the shell.

$ export something=foobar
$ export hello=world
$ /path/to/command

Exec will timeout in 1 hour and report as failure.

examples:

  • exec:./check.exe
  • exec:/usr/local/bin/check.sh
Extra report output for exec

In exec, you can set latency of service, and status of service with the output of the command. Please write output like below.

::latency::123.456
::status::failure
hello world

This output is reporting latency is 123.456ms, status is FAILURE, and message is hello world.

  • ::latency::: Reports the latency of service in milliseconds.
  • ::status::: Reports the status of service in healthy, degrade, failure, aborted, or unknown.

Ayd uses the last value if found multiple reports in single output.

for Alert

In alerting, Ayd sets some environment variables about the incident. The name of variable and meaning is the same as the queries of HTTP scheme's alerting.

source:

This is a special scheme for load targets from a file. Load each line in the file as a target URL and check all targets.

Source file is looks like below.

# servers
ping:somehost.example.com
ping:anotherhost.example.com
ping:yet.anotherhost.example.com

# services
https://service1.example.com
https://service2.example.com

# you can also read another file
source:./another-list.txt

The line that starts with # will ignored as a comment.

Source file should encoded by UTF-8 with/without BOM or UTF-16 with BOM, but in Windows, you can use legacy encoding. Please see also text encoding chapter.

examples:

  • source:./targets.txt
  • source:/path/to/targets.txt
source+http: / source+https:

source+http: and source+https: is variants of source:. These fetch HTTP/HTTPS and load the response as a source file.

WARNING: Please don't use it if you can't completely trust the HTTP server because this scheme can be a backdoor of your server. For example, Ayd will execute everything even if HTTP server responses exec:rm#/your/important/directory

examples:

  • source+https://example.com/targets.txt
source+ftp: / source+ftps:

source+ftp: and source+ftps: is variants of source: that very similar to source+http:. These download source file via FTP/FTPS and load it.

WARNING: Please don't use it if you can't completely trust the source file in the FTP server because this scheme can be a backdoor of your server. For example, Ayd will execute everything even if the FTP server responses exec:rm#/your/important/directory

examples:

  • source+ftp://example.com/targets.txt
source+exec:

source+exec: is another variant of source:. It is execute script as the same way as exec: and load the output as a source file.

examples:

  • source+exec:./make-targets-list.exe
  • source+exec:/usr/local/bin/targets.sh
plugin

Plugin is a executable file named like ayd-xxx-probe or ayd-xxx-alert, and installed to the PATH directory.

For example, you can use xxx: scheme if you have installed a executable file named ayd-xxx-probe. Of course, you can change executable file name to change scheme name.

If you want to make your own plugin please read make plugin section.

Ayd expects UTF-8 text as outputs of plugins. But in Windows, you can use system's default character encoding. Please see also text encoding chapter.

Plugin will timeout in maximum 1 hour and report as failure.

plugin list
for Probe
for Alert

Scheduling

In default, Ayd will check targets every 5 minutes.

If you want, You can place the schedule specifications before the target URLs like below.

$ ayd 10m https://your-service.example.com 1h https://another-service.example.com

The above command means checking your-service.example.com every 10 minutes, and checking another-service.example.com every 1 hour.

You can also use the Cron style spec as a schedule spec like below.

$ ayd '*/5  6-21 * *'     https://your-service.example.com \
      '*/10 *    * * 1-5' https://another-service.example.com

The above command means checking your-service.example.com every 5 minutes from 6 a.m. to 9 p.m, and checking another-service.example.com every 10 minutes from monday to friday.

 ┌─────── minute (0 - 59)
 │ ┌────── hour (0 - 23)
 │ │ ┌───── day of the month (1 - 31)
 │ │ │ ┌──── month (1 - 12)
 │ │ │ │ ┌─── [optional] day of the week (0 - 6 (sunday - saturday))
 │ │ │ │ │
'* * * * *'

Log file

The log file of Ayd is stored in TSV (Tab Separated Values) format, encoded UTF-8. The log has these columns.

  1. Timestamp in RFC3339 format like 2001-02-30T16:05:06+00:00.

  2. Status of the record that HEALTHY, DEGRADE, FAILURE, UNKNOWN, or ABORTED.

    • HEALTHY means service seems working well.

    • DEGRADE means service seems working but partially degraded. You should do something to the target system because the target is not completely healthy.

    • FAILURE means service seems failure or stopped. You should do something to the target system because the target may be broken if received this status.

    • UNKNOWN means Ayd is failed to status checking. For example, not found test script, failed to resolve service name, etc. You should check the target system, other systems like DNS, or Ayd settings because maybe something worse happened if received this status.

    • ABORTED means Ayd terminated during status checking. For example, Ayd reports this when terminated Ayd with Ctrl-C. You do not have to action about this status because it happens by your operation. (might be you have to check Ayd settings if you do not know why caused this)

  3. Latency of the service in milliseconds.

    Some probes like ping: reports average latency, and other probes reports total value..

  4. Target URL.

    This URL is the same to passed one as argument, but normalized. For example, ping:somehost?hello=world to be ping:somehost because ping: does not use query values.

  5. The detail of status, the reason for failure, or the output of the executed script.

For example, log lines look like below.

2001-02-30T16:00:00+09:00	FAILURE	0.544	http://localhost	Get "http://localhost": dial tcp [::1]:80: connect: connection refused
2001-02-30T16:05:00+09:00	UNKNOWN	0.000	tcp:somehost:1234	lookup somehost on 192.168.1.1:53: no such host
2001-02-30T16:10:00+09:00	HEALTHY	0.375	ping:anotherhost	rtt(min/avg/max)=0.31/0.38/0.47 send/rcv=4/4

Ayd will save the log file named ayd.log into the current directory in default. You can change this with -f option like below.

$ ayd -f /path/to/ayd.log ping:example.com

There is no feature to log rotate. Please consider using the log rotation tool if you have a plan to use it for a long time. (Ayd can handle the huge log, but it is not easy to investigate the huge log when trouble)

Please use -f - option for disable writing log file if you don't use log file. This is not recommended for production use because Ayd can't restore last status when restore if don't save log file. But, this is may useful for use Ayd as a parts of script file.

In old version, before 0.10.0 or older, the -f option was named -o option. The -o option is still working in the latest version, but it will removed in the future version.

If you want log file in other format, you can download in CSV or JSON via HTTP endpoint. Or, you can use ayd conv subcommand like below.

$ cat ayd.log | ayd conv > ayd_log.csv

$ ayd conv ./ayd.log -o ayd_log.csv

$ ayd conv -j ./ayd.log -o ayd_log.json

$ ayd conv -J ./ayd.log -o ayd_log.jsonl

Alerting

Ayd can do something for alerting when the target status changed like an incident caused or recovered. The alert is specified as a URL same as a target URL.

The alert URL specify looks like below.

$ ayd -a https://alert.example.com/alert https://target.example.com

You can use -a option more than once if you want Ayd to send alert to multiple systems like below.

$ ayd -a exec:./alert.sh -a mailto:admin@example.com https://target.example.com
plugins
e-mail (SMTP)

The ayd-mailto-alert is a plugin to sending an alert e-mail via SMTP.

The screenshot of Ayd alert in email. You can see service status, target URL, and reason to failure. And there is button to open Status Page.

You can use this plugin like below.

$ export SMTP_SERVER=smtp.example.com:465 SMTP_USERNAME=your-name SMTP_PASSWORD=your-password
$ export AYD_URL="http://ayd-external-url.example.com"

$ ayd -a mailto:your-email@example.com https://target.example.com

Please see more information in the readme of ayd-mailto-alert.

Slack

The ayd-slack-alert is a plugin to sending an alert to Slack.

The screenshot of Ayd alert in the Slack. You can see service status, target URL, and reason to failure. And there is button to open Status Page.

You can use this plugin like below.

$ export SLACK_WEBHOOK_URL="https://hooks.slack.com/services/......"
$ export AYD_URL="http://ayd-external-url.example.com"

$ ayd -a slack: https://target.example.com

Please see more information in the readme of ayd-slack-alert.

Daemonize

Use docker

You can use docker image for execute Ayd. This image includes ayd, and alert plugin for email and slack.

$ docker run --restart=always -v /var/log/ayd:/var/log/ayd macrat/ayd http://your-target.example.com

Of course, you can also use docker-compose or Kubernetes, etc. Please see ayd-docker repository for more information about this contianer image.

Systemd

If you using systemd, it is easy to daemonize Ayd.

Please put ayd command to /usr/local/bin/ayd (you can use another place if you want), and write a setting like below to /etc/systemd/system/ayd.service.

[Unit]
Description=Ayd status monitoring server
After=network.target remote-fs.target

[Service]
ExecStart=/usr/local/bin/ayd -f /var/log/ayd.log \
    http://your-target.example.com
#   ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ please change target

[Install]
WantedBy=multi-user.target

And then, you can enable this service.

# reload config
$ sudo systemctl daemon-reload

# start service
$ sudo systemctl start ayd

# enable auto start when boot system
$ sudo systemctl enable ayd

Make plugin

Plugins in Ayd is a executable file named like ayd-xxx-probe or ayd-xxx-alert, and installed to the PATH directory.

Ayd looks for ayd-xxx-probe for a target or ayd-xxx-alert for an alert, if URL scheme is xxx:, xxx-yyy:, or xxx+yyy:. You can change scheme via changing xxx, but you can't use URL schemes that ayd, alert, and the scheme that is supported by Ayd itself. And you can use yyy part to change plugin behavior, the same as http: or dns:.

The longest plugin name has priority if you installed multiple plugins. For example, ayd-xxx-yyy-probe has priority than ayd-xxx-probe.

The output of the plugin will parsed the same way to log file.

The differences from plugin to exec: are below.

exec: plugin
URL Scheme exec: only anything
executable file directory anywhere only in the PATH directory
set argument and environment variable in URL can can not
receive raw target URL can not can
record about multiple targets like as source can not can

There is a library for create plugin.

Probe plugin

Probe plugin receives target URL as the first argument of the command.

For example, target URL foobar:your-target has the same mean to below command.

ayd-foobar-probe "foobar:your-target"
Alert plugin

Alert plugin receives the URL of alert at 1st of argument, and 2nd or after arguments is the same values and order as the log file.

For example, the alert URL foobar:your-alert for plugin ayd-foobar-alert called like a below command.

$ ayd-foobar-alert              \
    "foobar:your-alert"         \
    "2001-02-30T16:05:06+09:00" \
    "FAILURE"                   \
    "1.234"                     \
    "ping:your-target"          \
    "this is message of the record"

The output of the probe plugin will parsed the same way to log file, but all target URL will add alert: prefix and won't not show in status page.

Text encoding

Ayd expects UTF-8 with/without BOM or UTF-8 with BOM as input character encoding.

But in Windows, you can use the system's default character encoding too, for example CP1252 or CP932. Ayd tries to decode as UTF-8 first, and then tries to use the system's default encoding. If the text has the BOM, Ayd always follow it.

The characters couldn't decode will replaced by U+FFFD that means unrecognized character before save to the log file. That means;

  • The log file is always valid UTF-8 even if your external command or plugin writes invalid characters.
  • You can lose information if external commands or plugins write invalid characters as current encoding.

Other options

Change listen port

You can change the HTTP server listen port with -p option. In default, Ayd uses port 9000.

Use HTTPS

You can set certificate file and key file via -c option and -k option.

$ ayd -c ./your-certificate.crt -k ./your-certificate.key ping:localhost

The HTTP/2 will be enabled if set these options.

Enable authentication for status pages

Ayd has very simple authentication mechanism using Basic Authentication. You can use it like below.

$ ayd -u user:p@ssword ping:localhost

For above example, you can access status page using user as username and p@ssword as password.

This is not very secure because you have to write a plain password to argument. (Attacker can peek arguments of other process easily if you have access to the server terminal) But, this is very easy to setup, and at least, work well against end user who don't know how to attack. If you need more secure option, please consider use reverse proxy like Nginx.

One-shot mode

If you want to use Ayd in a script, you may use -1 option. Ayd will check status just once and exit when passed -1 option.

Exit status code is 0 if all targets are healthy. If some targets are unhealthy, the status code is 1. And, if your arguments are wrong (or can't resolve host names, or exec scripts not found), the status code is 2.

Documentation

The Go Gopher

There is no documentation for this package.

Directories

Path Synopsis
Package internal is internal packages for Ayd.
Package internal is internal packages for Ayd.
ayderr
Package ayderr is the set of error types in Ayd.
Package ayderr is the set of error types in Ayd.
endpoint
Package endpoint implements the HTTP endpoints of Ayd such as status page.
Package endpoint implements the HTTP endpoints of Ayd such as status page.
scheme
Package scheme implements how to probe or send alert to the URL.
Package scheme implements how to probe or send alert to the URL.
scheme/textdecode
Package textdecode is a utility package to decode text.
Package textdecode is a utility package to decode text.
store
Package store implements how to output the log, and how to restore from the log file.
Package store implements how to output the log, and how to restore from the log file.
testutil
Package testutil implements utilities for testing.
Package testutil implements utilities for testing.
The library for making Ayd plugin or client.
The library for making Ayd plugin or client.

Jump to

Keyboard shortcuts

? : This menu
/ : Search site
f or F : Jump to
y or Y : Canonical URL