Skip to content

Commit 990656c

Browse files
committed
Merge branch 'release/0.4.0' into stable
2 parents 27c8b3b + 36b3ee5 commit 990656c

File tree

22 files changed

+301
-68
lines changed

22 files changed

+301
-68
lines changed

README.md

+64-9
Original file line numberDiff line numberDiff line change
@@ -15,14 +15,14 @@ add additional support.
1515
```elixir
1616
def deps do
1717
[{:distillery, "~> 1.3",
18-
{:bootleg, "~> 0.3"}]
18+
{:bootleg, "~> 0.4"}]
1919
end
2020
```
2121

2222
## Build server setup
2323

2424
In order to build your project, Bootleg requires that your build server be set up to compile
25-
Elixir code. Make sure you have already installed Elixir on any build server you define. The remote
25+
Elixir code. Make sure you have already installed Elixir on any build server you define.
2626

2727

2828
## Quick Start
@@ -42,8 +42,8 @@ $ mix bootleg.init
4242
# config/deploy.exs
4343
use Bootleg.Config
4444

45-
role :build, "your-build-server.local", user: "develop", password: "bu1ldm3", workspace "/some/build/workspace"
46-
role :app, ["web1", "web2", "web3"], user: "admin", password: "d3pl0y", workspace "/var/myapp"
45+
role :build, "your-build-server.local", user: "develop", password: "bu1ldm3", workspace: "/some/build/workspace"
46+
role :app, ["web1", "web2", "web3"], user: "admin", password: "d3pl0y", workspace: "/var/myapp"
4747
```
4848

4949
### build and deploy
@@ -68,6 +68,52 @@ role :build, "build.example.com", user, "build", port: "2222", workspace: "/tmp/
6868
role :app, ["web1.example.com", "web2.myapp.com"], user: "admin", workspace: "/var/www/myapp"
6969
```
7070

71+
### Environments
72+
73+
Bootleg has its own concept of environments, which is analogous to but different from `MIX_ENV`. Bootleg environments
74+
are used if you have multiple clusters that you deploy your code to, such as a QA or staging cluster, in addition to
75+
your `production` cluster. Your main Bootleg config still goes in `config/deploy.exs`, and environment specific details
76+
goes in `config/deploy/your_bootleg_env.exs`. The selected environment config file gets loaded immediately after
77+
`config/deploy.exs`. To invoke a Bootleg command with a specific environment, simply pass it as the first argument to
78+
any bootleg Mix command.
79+
80+
For example, say you have both a `production` and a `staging` cluster. Your configuration might look like:
81+
82+
```elixir
83+
# config/deploy.exs
84+
use Bootleg.Config
85+
86+
task :my_nifty_thing do
87+
Some.jazz()
88+
end
89+
90+
after_task :deploy, :my_nifty_thing
91+
92+
role :build, "build.example.com", user, "build", port: "2222", workspace: "/tmp/build/myapp"
93+
```
94+
95+
```elixir
96+
# config/deploy/production.exs
97+
use Bootleg.Config
98+
99+
role :app, ["web1.example.com", "web2.example.com"], user: "admin", workspace: "/var/www/myapp"
100+
```
101+
102+
```elixir
103+
# config/deploy/staging.exs
104+
use Bootleg.Config
105+
106+
role :app, ["stage1.example.com", "stage2.example.com"], user: "admin", workspace: "/var/www/myapp"
107+
```
108+
109+
110+
Then if you wanted to update staging, you would `mix bootleg.update staging`. If you wanted to update production,
111+
it would be `mix bootleg.update production`, or just `mix bootleg.update` (the default environment is `production`).
112+
113+
It is not a requirement that you define an environment file for each environment, but you will get a warning if
114+
a specific environment file can't be found. It is strongly encouraged to have an environment file per environment.
115+
116+
71117
## Roles
72118

73119
Actions in Bootleg are paired with roles, which are simply a collection of hosts that are responsible for the same function, for example building a release, archiving a release, or executing commands against a running application.
@@ -135,7 +181,7 @@ Bootleg extensions may impose restrictions on certain roles, such as restricting
135181

136182
* `build` - Takes only one host. If a list is given, only the first hosts is
137183
used and a warning may result. If this role isn't set the release packaging will be done locally.
138-
* `app` - Takes a lists of hosts, or a string with one host.
184+
* `app` - Takes a list of hosts, or a string with one host.
139185

140186
## Building and deploying a release
141187

@@ -154,6 +200,14 @@ mix bootleg.update production
154200
Note that `bootleg.update` will stop any running nodes and then perform a cold start. The stop is performed with
155201
the task `stop_silent`, which differs from `stop` in that it does not fail if the node is already stopped.
156202

203+
`bootleg.build` will clean the remote workspace prior to copying the code over, to ensure that any files left from
204+
a previous build do not cause issues. The entire contents of the remote workspace are removed via `rm -rf *` from
205+
the root of the workspace. You can configure this behavior by setting the config option `clean_locations`, which
206+
takes a list of locations and passes them to `rm -rf` on the remote server. Relative paths will be interpreted relative
207+
to the workspace, absolute paths will be treated as is. Warning: this means that `config :clean_locations, ["/"]` would
208+
attempt to erase the entire root file system of your remote server. Be careful when altering `clean_locations` and never
209+
use a privileged user on your build server.
210+
157211
## Admin Commands
158212

159213
Bootleg has a set of commands to check up on your running nodes:
@@ -183,8 +237,9 @@ Hooks are defined within `config/deploy.exs`. Hooks may be defined to trigger
183237
before or after a task. The following tasks are provided by Bootleg:
184238

185239
1. `build` - build process for creating a release package
186-
1. `compile` - compilation of your project
187-
2. `generate_release` - generation of the release package
240+
1. `clean` - cleans the remote workspace
241+
2. `compile` - compilation of your project
242+
3. `generate_release` - generation of the release package
188243
2. `deploy` - deploy of a release package
189244
3. `start` - starting of a release
190245
4. `stop` - stopping of a release
@@ -344,7 +399,7 @@ for building phoenix releases.
344399
# mix.exs
345400
def deps do
346401
[{:distillery, "~> 1.3"},
347-
{:bootleg, "~> 0.3"},
402+
{:bootleg, "~> 0.4"},
348403
{:bootleg_phoenix, "~> 0.1"}]
349404
end
350405
```
@@ -353,7 +408,7 @@ For more about `bootleg_phoenix` see: https://github.com/labzero/bootleg_phoenix
353408

354409
## Sharing Tasks
355410

356-
Sharing is a good thing. We love to share, espically awesome code we write. Bootleg supports loading
411+
Sharing is a good thing. We love to share, especially awesome code we write. Bootleg supports loading
357412
tasks from packages in a manner very similar to `Mix.Task`. Just define your module under `Bootleg.Tasks`,
358413
`use Bootleg.Task` and pass it a block of Bootleg DSL. The contents will be discovered and executed
359414
automatically at launch.

lib/bootleg/config.ex

+13
Original file line numberDiff line numberDiff line change
@@ -497,6 +497,19 @@ defmodule Bootleg.Config do
497497
get_config(:version, Project.config[:version])
498498
end
499499

500+
@doc false
501+
@spec env() :: any
502+
def env do
503+
get_config(:env, :production)
504+
end
505+
506+
@doc false
507+
@spec env(any) :: :ok
508+
def env(env) do
509+
{:ok, _} = Bootleg.Config.Agent.start_link(env)
510+
config(:env, env)
511+
end
512+
500513
@doc false
501514
@spec split_roles_and_filters(atom | keyword) :: {[atom], keyword}
502515
defp split_roles_and_filters(role) do

lib/bootleg/config/agent.ex

+2-2
Original file line numberDiff line numberDiff line change
@@ -6,9 +6,9 @@ defmodule Bootleg.Config.Agent do
66
@typep data :: keyword
77

88
@spec start_link() :: {:ok, pid}
9-
def start_link do
9+
def start_link(env \\ :production) do
1010
state_fn = fn ->
11-
[roles: [], config: [], before_hooks: [], after_hooks: [], next_hook_number: 0]
11+
[roles: [], config: [env: env], before_hooks: [], after_hooks: [], next_hook_number: 0]
1212
end
1313
case Agent.start_link(state_fn, name: Bootleg.Config.Agent) do
1414
{:error, {:already_started, pid}} -> {:ok, pid}

lib/bootleg/mix_task.ex

+4-2
Original file line numberDiff line numberDiff line change
@@ -8,13 +8,15 @@ defmodule Bootleg.MixTask do
88

99
@spec run(OptionParser.argv) :: :ok
1010
if is_atom(unquote(task)) && unquote(task) do
11-
def run(_args) do
11+
def run(args) do
12+
Config.env(List.first(args) || :production)
1213
use Config
1314

1415
invoke unquote(task)
1516
end
1617
else
17-
def run(_args) do
18+
def run(args) do
19+
Config.env(List.first(args) || :production)
1820
:ok
1921
end
2022
end

lib/bootleg/tasks.ex

+9-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
defmodule Bootleg.Tasks do
22
@moduledoc false
3-
alias Bootleg.Config
3+
alias Bootleg.{Config, UI}
44

55
def load_tasks do
66
use Config
@@ -16,7 +16,14 @@ defmodule Bootleg.Tasks do
1616

1717
load_third_party()
1818

19-
Config.load("config/deploy.exs")
19+
Config.load(Path.join("config", "deploy.exs"))
20+
case Config.load(Path.join(["config", "deploy", "#{Config.env}.exs"])) do
21+
{:error, :enoent} -> UI.warn("You are running in the `#{Config.env}` bootleg " <>
22+
"environment but there is no configuration defined for that environment. " <>
23+
"Create one at `config/deploy/#{Config.env}.exs` if you want to do additional " <>
24+
"customization.")
25+
val -> val
26+
end
2027
:ok
2128
end
2229

lib/bootleg/tasks/build.exs

+14-2
Original file line numberDiff line numberDiff line change
@@ -7,17 +7,29 @@ end
77

88
task :generate_release do
99
UI.info "Generating release"
10-
mix_env = Bootleg.Config.get_config(:mix_env, "prod")
10+
mix_env = Keyword.get(Config.config(), :mix_env, "prod")
1111
remote :build do
1212
"MIX_ENV=#{mix_env} mix release"
1313
end
1414
end
1515

1616
task :compile do
17-
mix_env = Bootleg.Config.get_config(:mix_env, "prod")
17+
mix_env = Keyword.get(Config.config(), :mix_env, "prod")
1818
UI.info "Compiling remote build"
1919
remote :build do
2020
"MIX_ENV=#{mix_env} mix deps.compile"
2121
"MIX_ENV=#{mix_env} mix compile"
2222
end
2323
end
24+
25+
task :clean do
26+
locations = config()
27+
|> Keyword.get(:clean_locations, ["*"])
28+
|> List.wrap
29+
|> Enum.join(" ")
30+
if locations != "" do
31+
remote :build do
32+
"rm -rvf #{locations}"
33+
end
34+
end
35+
end

lib/mix/tasks/init.ex

+34-14
Original file line numberDiff line numberDiff line change
@@ -11,16 +11,19 @@ defmodule Mix.Tasks.Bootleg.Init do
1111

1212
def run(_args) do
1313
deploy_file_path = Path.join(["config", "deploy.exs"])
14+
deploy_dir_path = Path.join(["config", "deploy"])
15+
production_file_path = Path.join(deploy_dir_path, "production.exs")
1416
Generator.create_directory("config")
1517
Generator.create_file(deploy_file_path, deploy_file_text())
18+
Generator.create_directory(deploy_dir_path)
19+
Generator.create_file(production_file_path, production_file_text())
1620
end
1721

1822
Generator.embed_text(:deploy_file, """
1923
use Bootleg.Config
2024
2125
# Configure the following roles to match your environment.
2226
# `build` defines what remote server your distillery release should be built on.
23-
# `app` defines what remote servers your distillery release should be deployed and managed on.
2427
#
2528
# Some available options are:
2629
# - `user`: ssh username to use for SSH authentication to the role's hosts
@@ -29,21 +32,38 @@ defmodule Mix.Tasks.Bootleg.Init do
2932
# - `workspace`: remote file system path to be used for building and deploying this Elixir project
3033
3134
role :build, "build.example.com", workspace: "/tmp/bootleg/build"
32-
role :app, ["app1.example.com", "app2.example.com"], workspace: "/var/app/example"
3335
34-
# Phoenix has some extra build steps which can be defined as task after the compile step runs.
36+
# Phoenix has some extra build steps such as asset digesting that need to be done during
37+
# compilation. To have bootleeg handle that for you, include the additional package
38+
# `bootleg_phoenix` to your `deps` list. This will automatically perform the additional steps
39+
# required for building phoenix releases.
40+
#
41+
# ```
42+
# # mix.exs
43+
# def deps do
44+
# [{:distillery, "~> 1.3"},
45+
# {:bootleg, "~> 0.4"},
46+
# {:bootleg_phoenix, "~> 0.1"}]
47+
# end
48+
# ```
49+
# For more about `bootleg_phoenix` see: https://github.com/labzero/bootleg_phoenix
50+
51+
""")
52+
53+
Generator.embed_text(:production_file, """
54+
use Bootleg.Config
55+
56+
# Configure the following roles to match your environment.
57+
# `app` defines what remote servers your distillery release should be deployed and managed on.
3558
#
36-
# Uncomment the following task definition if this is a Phoenix application. To learn more about
37-
# hooks and adding additional behavior to your deploy workflow, please refer to the bootleg
38-
# README which can be found at https://github.com/labzero/bootleg/blob/master/README.md
39-
40-
# after_task :compile do
41-
# remote :build do
42-
# "[ -f package.json ] && npm install || true"
43-
# "[ -f brunch-config.js ] && [ -d node_modules ] && ./node_modules/brunch/bin/brunch b -p || true"
44-
# "[ -d deps/phoenix ] && mix phoenix.digest || true"
45-
# end
46-
# end
59+
# Some available options are:
60+
# - `user`: ssh username to use for SSH authentication to the role's hosts
61+
# - `password`: password to be used for SSH authentication
62+
# - `identity`: local path to an identity file that will be used for SSH authentication instead of a password
63+
# - `workspace`: remote file system path to be used for building and deploying this Elixir project
64+
65+
role :app, ["app1.example.com", "app2.example.com"], workspace: "/var/app/example"
66+
4767
""")
4868

4969
end

lib/strategies/build/distillery.ex

+1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@ defmodule Bootleg.Strategies.Build.Distillery do
1717

1818
mix_env = Config.get_config(:mix_env, "prod")
1919
refspec = Config.get_config(:refspec, "master")
20+
invoke :clean
2021
:ok = git_push(conn, refspec)
2122
git_reset_remote(conn, refspec)
2223
git_clean_remote(conn)

lib/ui.ex

+1
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,7 @@ defmodule Bootleg.UI do
5252
defp validate_verbosity(verbosity)
5353
defp validate_verbosity(:warning), do: :warning
5454
defp validate_verbosity(:debug), do: :debug
55+
defp validate_verbosity(:silent), do: :silent
5556
defp validate_verbosity(_), do: :info
5657

5758
defp verbosity_includes(setting, level)

mix.exs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
defmodule Bootleg.Mixfile do
22
use Mix.Project
33

4-
@version "0.3.0"
4+
@version "0.4.0"
55
@source "https://github.com/labzero/bootleg"
66

77
def project do

test/bootleg/config/agent_test.exs

+17-2
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ defmodule Bootleg.Config.AgentTest do
44

55
test "stores values for retrieval" do
66
{:ok, _} = Agent.start_link
7-
assert Agent.get(:config) == []
7+
assert Agent.get(:config) == [env: :production]
88

99
Agent.put(:config, [key: :value, key2: :value])
1010
assert Agent.get(:config) == [key: :value, key2: :value]
@@ -13,8 +13,23 @@ defmodule Bootleg.Config.AgentTest do
1313
assert Agent.get(:config) == [key: :value, key2: :value, foo: :bar]
1414
end
1515

16-
test "startlink/0 ignores 'already started' errors" do
16+
test "start_link/0 ignores 'already started' errors" do
1717
Agent.start_link
1818
assert {:ok, _} = Agent.start_link
1919
end
20+
21+
test "start_link/1 ignores 'already started' errors" do
22+
Agent.start_link
23+
assert {:ok, _} = Agent.start_link(:foo)
24+
end
25+
26+
test "start_link/0 sets the environment to `production`" do
27+
Agent.start_link
28+
assert Agent.get(:config) == [env: :production]
29+
end
30+
31+
test "start_link/1 sets the environment to the provided env" do
32+
Agent.start_link(:bar)
33+
assert Agent.get(:config) == [env: :bar]
34+
end
2035
end

0 commit comments

Comments
 (0)