From ce1900c47778d86a6f7b15da423898e5e7647743 Mon Sep 17 00:00:00 2001 From: Mitchell Hashimoto Date: Tue, 18 Jun 2013 14:36:21 -0700 Subject: [PATCH] website: custom post-processor dev --- .../source/docs/extend/command.html.markdown | 2 +- .../docs/extend/post-processor.html.markdown | 91 +++++++++++++++++++ website/source/layouts/docs.erb | 1 + 3 files changed, 93 insertions(+), 1 deletion(-) create mode 100644 website/source/docs/extend/post-processor.html.markdown diff --git a/website/source/docs/extend/command.html.markdown b/website/source/docs/extend/command.html.markdown index f1847ca9c..2c89e6f38 100644 --- a/website/source/docs/extend/command.html.markdown +++ b/website/source/docs/extend/command.html.markdown @@ -29,7 +29,7 @@ in the core Packer configuration. The interface that must be implemented for a command is the `packer.Command` interface. It is reproduced below for easy reference. The reference below -also contains some basic documentatin of what each of the methods are +also contains some basic documentation of what each of the methods are supposed to do.
diff --git a/website/source/docs/extend/post-processor.html.markdown b/website/source/docs/extend/post-processor.html.markdown
new file mode 100644
index 000000000..95a893132
--- /dev/null
+++ b/website/source/docs/extend/post-processor.html.markdown
@@ -0,0 +1,91 @@
+---
+layout: "docs"
+---
+
+# Custom Post-Processor Development
+
+Post-processors are the components of Packer that transform one artifact
+into another, for example by compressing files, or uploading them.
+
+In the compression example, the transformation would be taking an artifact
+with a set of files, compressing those files, and returning a new
+artifact with only a single file (the compressed archive). For the
+upload example, the transformation would be taking an artifact with
+some set of files, uploading those files, and returning an artifact
+with a single ID: the URL of the upload.
+
+Prior to reading this page, it is assumed you have read the page on
+[plugin development basics](/docs/extend/developing-plugins.html).
+
+Post-processor plugins implement the `packer.PostProcessor` interface and
+are served using the `plugin.ServePostProcessor` function.
+
+
+ Warning! This is an advanced topic. If you're new to Packer, + we recommend getting a bit more comfortable before you dive into writing + plugins. +
+ + +## The Interface + +The interface that must be implemented for a post-processor is the +`packer.PostProcessor` interface. It is reproduced below for easy reference. +The reference below also contains some basic documentation of what each of +the methods are supposed to do. + +
+// A PostProcessor is responsible for taking an artifact of a build
+// and doing some sort of post-processing to turn this into another
+// artifact. An example of a post-processor would be something that takes
+// the result of a build, compresses it, and returns a new artifact containing
+// a single file of the prior artifact compressed.
+type PostProcessor interface {
+	// Configure is responsible for setting up configuration, storing
+	// the state for later, and returning and errors, such as validation
+	// errors.
+	Configure(interface{}) error
+
+	// PostProcess takes a previously created Artifact and produces another
+	// Artifact. If an error occurs, it should return that error.
+	PostProcess(Artifact) (Artifact, error)
+}
+
+ +### The "Configure" Method + +The `Configure` method for each post-processor is called early in the +build process to configure the post-processor. The configuration is passed +in as a raw `interface{]`. The configure method is responsible for translating +this configuration into an internal structure, validating it, and returning +any errors. + +For decoding the `interface{]` into a meaningful structure, the +[mapstructure](https://github.com/mitchellh/mapstructure) library is +recommended. Mapstructure will take an `interface{}` and decode it into an +arbitrarily complex struct. If there are any errors, it generates very +human-friendly errors that can be returned directly from the configure +method. + +While it is not actively enforced, **no side effects** should occur from +running the `Configure` method. Specifically, don't create files, don't +create network connections, etc. Configure's purpose is solely to setup +internal state and validate the configuration as much as possible. + +`Configure` being run is not an indication that `PostProcess` will ever +run. For example, `packer validate` will run `Configure` to verify the +configuration validates, but will never actually run the build. + +### The "PostProcess" Method + +The `PostProcess` method is where the real work goes. PostProcess is +responsible for taking one `packer.Artifact` implementation, and transforming +it into another. + +When we say "transform," we don't mean actually modifying the existing +`packer.Artifact` value itself. We mean taking the contents of the artifact +and creating a new artifact from that. For example, if we were creating +a "compress" post-processor that is responsible for compressing files, +the transformation would be taking the `Files()` from the original artifact, +compressing them, and creating a new artifact with a single file: the +compressed archive. diff --git a/website/source/layouts/docs.erb b/website/source/layouts/docs.erb index bd760b8f5..76e64388b 100644 --- a/website/source/layouts/docs.erb +++ b/website/source/layouts/docs.erb @@ -61,6 +61,7 @@
  • Developing Plugins
  • Custom Builder
  • Custom Command
  • +
  • Custom Post-Processor
  • Custom Provisioner