hookworm

package module
v0.3.1 Latest Latest
Warning

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

Go to latest
Published: Sep 4, 2013 License: MIT Imports: 16 Imported by: 0

README

hookworm

GitHub & Travis hook receiving thingydoo.

Usage

Usage: hookworm-server [options] [key=value...]
  -D="": Working directory (scratch pad)
  -P="": PID file (only written if flag given)
  -T=30: Timeout for handler executables (in seconds)
  -W="": Worm directory that contains handler executables
  -a=":9988": Server address
  -d=false: Show debug output
  -github.path="/github": Path to handle Github payloads
  -rev=false: Print revision and exit
  -travis.path="/travis": Path to handle Travis payloads
  -version=false: Print version and exit
  -version+=false: Print version, revision, and build tags

Hookworm is designed to listen for GitHub and Travis webhook payloads and delegate handling to a pipeline of executables. In this way, the long-running server process stays smallish (~6MB) and any increase in memory usage at payload handling time is ephemeral, assuming the handler executables aren't doing anything silly.

An example invocation that uses the handler executables shipped with hookworm would look like this, assuming the hookworm repo has been cloned into /var/lib/hookworm:

mkdir -p /var/run/hookworm-main
hookworm-server -d \
  -D /var/run/hookworm-main \
  -W /var/lib/hookworm/worm.d \
  syslog=yes >> /var/log/hookworm-main.log 2>&1
Handler contract

Handler executables are expected to fulfill the following contract:

  • has one of the following file extensions: .js, .pl, .py, .rb, .sh
  • does not begin with . (hidden file)
  • accepts a positional argument of configure
  • accepts positional arguments of handle github
  • accepts positional arguments of handle travis
  • writes only the (potentially modified) payload to standard output
  • exits 0 on success
  • exits 78 on no-op (roughly ENOSYS)

It is up to the handler executable to decide what is done for each command invocation. The execution environment includes the HOOKWORM_WORKING_DIR variable, which may be used as a scratch pad for temporary files.

<interpreter> <handler-executable> configure

The configure command is invoked at server startup time for each handler executable, passing the handler configuration as a JSON object on the standard input stream. The configuration object is guaranteed to have all of the values provided as flags to hookworm-server.

Additionally, any key-value pairs provided as postfix arguments will be added to a worm_flags hash such as the syslog=yes argument given in the above example. Bare keys are assigned a JSON value of true. String values of true, yes, and on are converted to JSON true, and string values of false, no, and off are converted to JSON false.

<interpreter> <handler-executable> handle github

The handle github command is invoked whenever a payload is received at the GitHub-handling path (/github by default). The payload is passed to the handler executable as a JSON object on the standard input stream.

<interpreter> <handler-executable> handle travis

The handle travis command is invoked whenever a payload is received at the Travis-handling path (/travis by default). The payload is passed to the handler executable as a JSON object on the standard input stream.

Included handlers

Hookworm ships with the following handlers:

Hookworm Annotator

The annotator is responsible for adding fields to the incoming payloads so that subsequent handlers do not have to duplicate decision-making logic.

GitHub payload annotation

GitHub payloads are given the following additional fields dependending on the presence of certain options.

is_pr_merge

Is the payload the result of a pull request merge?

is_watched_branch

Is the payload for a branch that is "watched", depending on the presence of the watched_branches postfix keyword argument?

has_watched_path

Does the payload contain changes to a "watched" path, depending on the presence of the watched_paths postfix keyword argument?

Hookworm Logger

The logger is responsible for logging valid incoming requests, optionally logging to syslog if the syslog=true postfix option is provided. Log verbosity is higher if the -d debug flag is passed.

Hookworm Rogue Commit Handler

The rogue commit handler is specific to GitHub payloads. It will inspect a payload in the context of the given watched_branches and watched_paths and send a "rogue commit email" to the email recipients given in email_recipients to provide visibility roughly equivalent to those commits that result from pull request merges.

Because the rogue commit handler is affected by so many arguments, here they are again with more details about their associated behavior:

watched_branches

The watched_branches argument should be a comma-delimited list of regular expressions, e.g.: watched_branches='^master$,^release_[0-9]'. If a commit payload is received that was not the result of a pull request merge and the Hookworm Annotator handler has determined that the branch name matches any of the entries in watched_branches, then a rogue commit email will be sent.

watched_paths

The watched_paths argument should be a comma-delimited list of regular expressions, e.g.: watched_paths='.*\.(go|rb|py)$,bin/.*'. If a commit payload is received that was not the result of a pull request merge and the Hookworm Annotator handler has determined that one of the commits in the payload contains a path matching any of the entries in watched_paths, then a rogue commit email will be sent.

email_from_addr

The email_from_addr is the email address used as the From header and SMTP MAIL address when sending rogue commit emails, e.g.: email_from_addr='hookworm-noreply@company.example.com'.

email_recipients

The email_recipients argument should be a comma-delimited list of email addresses (without display name) used in the To header and SMTP RCPT addresses when sending rogue commit emails, e.g.: email_recipients='devs@company.example.com,project-distro+hookworm@partner.example.net'

email_uri

The email_uri argument should be a well-formed URI containing the SMTP hostname and port and potentially the username and password used for plain SMTP auth, e.g.: email_uri='smtp://hookworm:secret@mailhost.example.com:2025'

Documentation

Index

Constants

This section is empty.

Variables

View Source
var (
	// VersionString is the git description set via ldflags
	VersionString string
	// RevisionString is the git revision set via ldflags
	RevisionString string
	// BuildTags are the tags used at build time set via ldflags
	BuildTags string
)

Functions

func ServerMain

func ServerMain() int

ServerMain is the `main` entry point used by the `hookworm-server` executable

Types

type Handler

type Handler interface {
	HandleGithubPayload(string) (string, error)
	HandleTravisPayload(string) (string, error)
	SetNextHandler(Handler)
	NextHandler() Handler
}

Handler is the interface each pipeline handler must fulfill

func NewHandlerPipeline

func NewHandlerPipeline(cfg *HandlerConfig) (Handler, error)

NewHandlerPipeline constructs a linked-list-like pipeline of handlers, each responsible for passing control to the next if deemed appropriate.

type HandlerConfig

type HandlerConfig struct {
	Debug         bool         `json:"debug"`
	GithubPath    string       `json:"github_path"`
	ServerAddress string       `json:"server_address"`
	ServerPidFile string       `json:"server_pid_file"`
	TravisPath    string       `json:"travis_path"`
	WorkingDir    string       `json:"working_dir"`
	WormDir       string       `json:"worm_dir"`
	WormTimeout   int          `json:"worm_timeout"`
	WormFlags     *wormFlagMap `json:"worm_flags"`
}

HandlerConfig contains the bag of configuration poo used by all handlers

type Server

type Server struct {
	// contains filtered or unexported fields
}

Server implements ServeHTTP, parsing payloads and handing them off to the handler pipeline

func NewServer

func NewServer(cfg *HandlerConfig) (*Server, error)

NewServer builds a Server instance given a HandlerConfig

func (*Server) ServeHTTP

func (srv *Server) ServeHTTP(w http.ResponseWriter, r *http.Request)

Directories

Path Synopsis

Jump to

Keyboard shortcuts

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