gocharm

module
v0.0.0-...-02218b7 Latest Latest
Warning

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

Go to latest
Published: Mar 26, 2015 License: LGPL-3.0

README

Gocharm - write your Juju charms in Go.

Gocharm is a framework that makes it easy to write charms in Go, with the benefits that Go brings: static typing, minimal dependencies and fun!

See the hook package docs for documentation on the core Go API and the gocharm (tool docs for how to use the gocharm tool itself.

Its features include:

  • all hooks are automatically generated. No need to create a hook file ever again.

  • config.yaml and the relations in metadata.yaml are also automatically generated.

  • built in support for persistent state in hooks.

  • packages implementing common relation interface types.

  • support for using the deployed Go charm binary as the running charm service itself.

  • a choice between deployment in "binary only" mode or with all source included in the charm and a Go development environment installed to aid in hook debugging.

A simple example charm

This example assumes you already have a working Go development environment and $GOPATH set up. It also assumes a current working Juju environment.

Define a Go charm by creating a Go package containing the following code. Choose any name for the package. For this example, I'll name it github.com/juju/gocharm/example-charms/donothing and store the code in runhook.go.

package mycharm
import (
	"github.com/juju/gocharm/hook"
)
func RegisterHooks(r *hook.Registry) {
	// Here we will register all any code to be called
	// when the charm runs, and any relations or
	// configuration values defined by the charm.
}

Like the main function for main packages, the RegisterHooks function is the top entry point for a charm. This function will be called in two contexts, when the charm is built by the gocharm command, and when the charm actually executes as part of a deployed unit. Thus the code in RegisterHooks must not assume that it is part of a deployed unit (which goes also for any packages or code called by RegisterHooks).

This particular charm will do absolutely nothing when run, because it registers no hooks. It is a valid charm package nonetheless.

We also need a metadata.yaml file in the package directory to hold the description and summary (and other metadata files too, such as tags) of the charm. Unlike a normal charm metadata.yaml, this will not hold any relations declared by the charm - they will be added at build time.

name: do-nothing
summary: 'any example that does nothing at all'
description: |
    This example charm does nothing.

To deploy it as a charm, we'll need to install it in our local juju charm repository. This is configured by setting the $JUJU_REPOSITORY environment variable.

For example, in your shell:

$ export JUJU_REPOSITORY=$HOME/charms

You will need the gocharm command:

$ go get github.com/juju/gocharm/cmd/gocharm

Then, in your package directory, run the gocharm command:

$ cd $GOPATH/src/github.com/juju/gocharm/example-charms/gosimple
$ gocharm
local:trusty/do-nothing

This has now compiled the charm and deployed it to your local juju charm repository.

$ tree $JUJU_REPOSITORY/trusty/do-nothing
/home/rog/charms/trusty/do-nothing
├── bin
│   └── runhook
├── hooks
│   ├── install
│   └── start
├── metadata.yaml
└── src
    ├── github.com
    │   └── juju
    │       └── gocharm
    │           └── example-charms
    │               └── do-nothing
    │                   ├── metadata.yaml
    │                   └── runhook.go
    └── runhook
        └── runhook.go

9 directories, 7 files

Note that the source code for the charm (but not that of all dependencies) has been copied to the charm directory, which is now a valid root for a GOPATH directory. The compiled binary which will run the hooks has been installed into bin/runhook. The hooks directory has been created and populated with an install and a start hook, neither of which will do anything when run.

$ cat $JUJU_REPOSITORY/trusty/do-nothing/hooks/install
#!/bin/sh
set -ex

$CHARM_DIR/bin/runhook install

The charm is now ready to be deployed.

$ juju deploy local:trusty/do-nothing
Added charm "local:trusty/do-nothing-0" to the environment.

After a little while, we see that the charm has deployed OK:

% juju status do-nothing
environment: local
machines:
  "2":
    agent-state: started
    agent-version: 1.22-alpha1.1
    dns-name: 10.0.3.184
    instance-id: rog-local-machine-2
    series: trusty
    hardware: arch=amd64
services:
  do-nothing:
    charm: local:trusty/do-nothing-0
    exposed: false
    units:
      do-nothing/0:
        agent-state: started
        agent-version: 1.22-alpha1.1
        machine: "2"
        public-address: 10.0.3.184

Of course, since the charm does nothing, there's not much to observe that it's working correctly.

Let's change runhook.go to register some hooks and to send a log message so that we can see something from the charm.

The start of the charm package remains the same.

// Package mycharm implements the simplest possible Go charm.
// It does nothing at all when deployed.
package mycharm

import (
	"github.com/juju/gocharm/hook"
)

We declare a type for the charm implementation. Although not strictly necessary, this is a common pattern. When the charm is running a hook, the ctxt field will hold the running hook context.

type nothing struct {
	ctxt *hook.Context
}

We change the RegisterHooks function so that it actually registers some things. We declare an instance of the nothing type (there will actually only ever be one instance of this type at runtime), and register some of its methods to be called when the hook runs.

func RegisterHooks(r *hook.Registry) {
	var n nothing
	r.RegisterContext(n.setContext, nil)
	r.RegisterHook("install", n.hook)
	r.RegisterHook("start", n.hook)
	r.RegisterHook("config-changed", n.hook)
}

When the hook runs, this setContext method is called first. It lets the charm code know about the running hook context, which enables it to use all the usual charm tools and enquire about configuration and relation settings.

func (n *nothing) setContext(ctxt *hook.Context) error {
	n.ctxt = ctxt
	return nil
}

Here is the actual hook code that runs in response to the hook. We just log a message to show that something is happening:

func (n *nothing) hook() error {
	n.ctxt.Logf("hook %s is doing nothing at all", n.ctxt.HookName)
	return nil
}

Having saved the file, we run gocharm again, and we can see that another hook file has been generated because we registered the "config-changed" hook.

$ gocharm
$  tree $JUJU_REPOSITORY/trusty/do-nothing/hooks
/home/rog/charms/trusty/do-nothing/hooks
├── config-changed
├── install
└── start

0 directories, 3 files

We can upgrade the existing service in place:

$ juju upgrade-charm do-nothing
Added charm "local:trusty/do-nothing-1" to the environment.

After a little while we can verify that the charm has changed:

$  juju ssh do-nothing/0 'sudo grep "doing nothing" /var/log/juju/unit-do-nothing-0.log'
Warning: Permanently added '10.0.3.184' (ECDSA) to the list of known hosts.
2014-11-25 13:08:46 INFO unit.do-nothing/0.juju-log cmd.go:247 hook config-changed is doing nothing at all
Connection to 10.0.3.184 closed.

Note that because we upgraded the charm, neither the install or start hooks are called again - only the config-changed hook.

TODO using persistent state

TODO importing relation packages

TODO using service package.

Directories

Path Synopsis
charmbits
elasticsearchrelation
The elasticsearchrelation package implements a Juju elasticsearch relation.
The elasticsearchrelation package implements a Juju elasticsearch relation.
httprelation
The httprelation package can be used in a charm that uses a relation with interface type "http".
The httprelation package can be used in a charm that uses a relation with interface type "http".
httpservice
The httpservice package provides a way to run a simple HTTP server service as a charm.
The httpservice package provides a way to run a simple HTTP server service as a charm.
mongodbrelation
The mongodbrelation package implements a Juju mongodb relation.
The mongodbrelation package implements a Juju mongodb relation.
service
The service packge provides a way for a charm to start and stop a service that runs independently of the charm hooks.
The service packge provides a way for a charm to start and stop a service that runs independently of the charm hooks.
simplerelation
The simplerelation package implements a generic simple relation where the provider units each make a set of attributes available, and the requirer units can see those attributes.
The simplerelation package implements a generic simple relation where the provider units each make a set of attributes available, and the requirer units can see those attributes.
cmd
gocharm
Gocharm processes a Go package ("." by default) and installs it as a Juju charm.
Gocharm processes a Go package ("." by default) and installs it as a Juju charm.
example-charms
concat
The concat package implements a charm that takes all the string values from units on upstream relations, concatenates them and makes them available to downstream relations.
The concat package implements a charm that takes all the string values from units on upstream relations, concatenates them and makes them available to downstream relations.
do-nothing
Package do-nothing implements the simplest possible Go charm.
Package do-nothing implements the simplest possible Go charm.
helloworld
The helloworld package implements an example charm that exposes an HTTP service that returns "hello, world" from every endpoint.
The helloworld package implements an example charm that exposes an HTTP service that returns "hello, world" from every endpoint.
helloworld-configurable
The helloworld-configurable package implements an example charm similar to http://godoc.org/github.com/juju/gocharm/example-charms/helloworld but which allows the message to be configured, demonstrating how changing configuration options can affect a running service.
The helloworld-configurable package implements an example charm similar to http://godoc.org/github.com/juju/gocharm/example-charms/helloworld but which allows the message to be configured, demonstrating how changing configuration options can affect a running service.
mongodbclient
The mongodbclient package implements an example charm that acts as the client of the mongodb charm.
The mongodbclient package implements an example charm that acts as the client of the mongodb charm.
The hook package provides a Go interface to the Juju charm hook commands.
The hook package provides a Go interface to the Juju charm hook commands.
hooktest
Package hooktest contains utilities for testing gocharm hooks.
Package hooktest contains utilities for testing gocharm hooks.
vendored

Jump to

Keyboard shortcuts

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