This file is written in reverse chronological order, newer releases will appear at the top.
- Add your entries here, remember to credit yourself however you want to be credited!
- Connection pooling. SSH connections are reused across multiple invocations
of
on()
, which can result in significant performance gains. See: capistrano#70. Matt @mbrictson Brictson. - Fixes to the Formatter::Dot and to the formatter class name resolver. @hab287:w
- Added the license to the Gemspec. @anatol.
- Fix :limit handling for the
in: :groups
run mode. Phil @phs Smith - Doc fixes @seanhandley, @sergey-alekseev.
- Support picking up a project local SSH config file, if a SSH config file exists at ./.ssh/config it will be merged with the ~/.ssh/config. This is ideal for defining project-local proxies/gateways, etc. Thanks to Alex @0rca Vzorov.
- Tests and general improvements to the Printer backends (mostly used internally). Thanks to Michael @miry Nikitochkin.
- Update the net-scp dependency version. Thanks again to Michael @miry Nikitochkin.
- Improved command map. This feature allows mapped variables to be pushed and unshifted onto the mapping so that the Capistrano extensions for rbenv and bundler, etc can work together. For discussion about the reasoning see capistrano/capistrano#639 and capistrano#45. A big thanks to Kir @kirs Shatrov.
test()
andcapture()
now behave as expected inside arun_locally
block meaning that they now run on your local machine, rather than erring out. Thanks to Kentaro @kentaroi Imai.- The
:wait
option is now successfully passed to the runner now. Previously the:wait
option was ignored. Thanks to Jordan @jhollinger Hollinger for catching the mistake in our test coverage. - Fixes and general improvements to the
download()
method which until now was quite naïve. Thanks to @chqr.
- Please see the Git history.
git rebase
ate our changelog (we should have been more careful)
- The gem now supports a run_locally, although it's nothing to do with SSH, it makes a nice API. There are examples in the EXAMPLES.md.
- Allow the setting of global SSH options on the
backend.ssh
as a hash, these options are the same as Net::SSH configure expects. Thanks to Rafał @lisukorin Lisowski
- Lots of small changes since 0.0.27.
- Particularly working around a possible NaN issue when uploading comparatively large files.
- Don't clobber SSH options with empty values. This allows Net::SSH to do the right thing most of the time, and look into the SSH configuration files.
- Pretty output no longer prints white text. ("Command.....")
- Fixed a double-output bug, where upon receiving the exit status from a remote command, the last data packet that it sent would be re-printed by the pretty formatter.
- Integration tests now use an Ubuntu Precise 64 Vagrant base box.
- Additional host declaration syntax,
SSHKit::Host
can now take a hash of host properties in addition to a number of new (common sense) DSN type syntaxes. - Changes to the constants used for logging, we no longer re-define a
Logger::TRACE
constant on the globalLogger
class, rather everyhing now usesSSHKit::Logger
(Thanks to Rafa Garcia) - Various syntax and documentation fixes.
upload!
anddownload!
now log to different levels depending on completion percentage. When the upload is 0 percent complete or a number indivisible by 10, the message is logged toLogger::DEBUG
otherwise the message is logged toLogger::INFO
, this should mean that normal users at a sane log level should see upload progress jump to100%
for small files, and otherwise for larger files they'll see output every10%
.
-
Pretty output now streams stdout and stderr. Previous versions would append (
+=
) chunks of data written by the remote host to theCommand
instance, and thePretty
formatter would only print stdout/stderr if the command was#complete?
. Historically this lead to issues where the remote process was blocking for input, had written the prompt to stdout, but it was not visible on the client side.Now each time the command is passed to the output stream, the stdout/stderr are replaced with the lines returned from the remote server in this chunk. (i.e were yielded to the callback block). Commands now have attribute accessors for
#full_stdout
and#full_stderr
which are appended in the way that#stdout
and#stderr
were previously.This should be considered a private API, and one should beware of relying on
#full_stdout
or#full_stderr
, they will likely be replaced with a cleaner soltion eventually. -
upload!
anddownload!
now print progress reports at theLogger::INFO
verbosity level.
- Explicitly rely on
net-scp
gem.
-
Added naïve implementations of
upload!()
anddownload!()
(syncoronous) to the Net::SSH backend. SeeEXAMPLES.md
for more extensive usage examples.The
upload!()
method can take a filename, or anIO
, this reflects the way the underlying Net::SCP implementation works. The same is true ofdownload!()
, when called with a single argument it captures the file's contents, otherwise it downloads the file to the local disk.on hosts do |host| upload!(StringIO.new('some-data-here'), '~/.ssh/authorized_keys') upload!('~/.ssh/id_rsa.pub', '~/.ssh/authorized_keys') puts download!('/etc/monit/monitrc') download!('/etc/monit/monitrc', '~/monitrc') end
-
Fixed an issue with default formatter
-
Modified
SSHKit.config.output_verbosity=
to accept different objects:SSHKit.config.output_verbosity = Logger::INFO SSHKit.config.output_verbosity = :info SSHKit.config.output_verbosity = 1
- Fixed a bug where the log level would be assigned, not compared in the pretty formatter, breaking the remainder of the output verbosity.
-
Modified the
Pretty
formatter to include the log level in front of executed commands. -
Modified the
Pretty
formatter not to print stdout and stderr by default, the log level must be raised to Logger::DEBUG to see the command outputs. -
Modified the
Pretty
formatter to useCommand#to_s
when printing the command, this prints the short form (without modifications/wrappers applied to the command for users, groups, directories, umasks, etc).
-
Enable
as()
to take either a string/symbol as previously, but also now accepts a hash of{user: ..., group: ...}
. In case that your host system supports the commandsg
(man 1 sg
) to switch your effective group ID then one can work on files as a team group user.on host do |host| as user: :peter, group: griffin do execute :touch, 'somefile' end end
will result in a file with the following permissions:
-rw-r--r-- 1 peter griffin 0 Jan 27 08:12 somefile
This should make it much easier to share deploy scripts between team members.
Note:
sg
has some very strict user and group password requirements (the user may not have a password (passwd username -l
to lock an account that already has a password), and the group may not have a password.)Additionally, and unsurprisingly the user must also be a member of the group.
sg
was chosen overnewgrp
as it's easier to embed in a one-liner command,newgrp
could be used with a heredoc, but my research suggested that it might be better to use sg, as it better represents my intention, a temporary switch to a different effective group. -
Fixed a bug with environmental variables and umasking introduced in 0.0.14. Since that version the environmental variables were being presented to the umask command's subshell, and not to intended command's subshell.
incorrect: `ENV=var umask 002 && env` correct: `umask 002 && ENV=var env`
-
Changed the exception handler, if a command returns with a non-zero exit status then the output will be prefixed with the command name and which channel any output was written to, for example:
Command.new("echo ping; false") => echo stdout: ping echo stderr: Nothing written
In this contrived example that's more or less useless, however with badly behaved commands that write errors to stdout, and don't include their name in the program output, it can help a lot with debugging.
-
Fixed a bug introduced in 0.0.16 where the capture() helper returned the name of the command that had been run, not it's output.
-
Classify the pre-directory switch, and pre-user switch command guards as having a DEBUG log level to exclude them from the logs.
- Fixed a bug introduced in 0.0.15 where the capture() helper returned boolean, discarding any output from the server.
-
Command
now takes averbosity
option. This defaults toLogger::INFO
and can be set to any of the Ruby logger level constants. You can also set it to the symbol:debug
(and friends) which will be expanded into the correct constants.The log verbosity level is set to Logger::INFO by default, and can be overridden by setting
SSHKit.config.output_verbosity = Logger::{...}
, pick a level that works for you.By default
test()
andcapture()
calls are surpressed, and not printed by the pretty logger as of this version.
- Umasks can now be set on
Command
instances. It can be set globally withSSHKit.config.umask
(default, nil; meaning take the system default). This can be used to set, for example a umask of007
for allowing users with the same primary group to share code without stepping on eachother's toes.
- Correctly quote
as(user)
commands, previously it would expand to:sudo su user -c /usr/bin/env echo "Hello World"
, in which the command to run was taken as simply/usr/bin/env
. By quoting all arguments it should now work as expected.sudo su user -c "/usr/bin/env echo \""Hello World\""
- Also print anything the program wrote to stdout when the exit status is non-zero and the command raises an error. (assits debugging badly behaved programs that fail, and write their error output to stdout.)
- Implementing confuguration objects on the backends (WIP, undocumented)
- Implement
SSHKit.config.default_env
, a hash which can be modified and will act as a globalwith
. - Fixed #9 (with(a: 'b', c: 'c') being parsed as
A=bC=d
. Now properly space separated. - Fixed #10 (overly aggressive shell escaping), one can now do: `with(path: 'foo:$PATH') without the $ being escaped too early.
- Include more attributes in
Command#to_hash
.
- Include more attributes in
Command#to_hash
.
- Added DSL method
background()
this sends a task to the background usingnohup
and redirects it's output to/dev/null
so as to avoid littering the filesystem withnohup.out
files.
Note: Backgrounding a task won't work as you expect if you give it a
string, that is you must do background(:sleep, 5)
and not background("sleep 5")
according to the rules by which a command is not processed in any way if it
contains a spaca character in it's first argument.
Usage Example:
on hosts do
background :rake, "assets:precompile" # typically takes 5 minutes!
end
Further: Many programs are badly behaved and no not work well with nohup
it has to do with the way nohup works, reopening the processe's file
descriptors and redirecting them. Programs that re-open, or otherwise
manipulate their own file descriptors may lock up when the SSH session is
disconnected, often they block writing to, or reading from stdin/out.
- DSL method
execute()
will now raiseSSHKit::Command::Failed
when the exit status is non-zero. The message of the exception will be whatever the process had written to stdout. - New DSL method
test()
behaves asexecute()
used to until this version. Command
now raises an error in#exit_status=()
if the exit status given is not zero. (see below)- All errors raised by error conditions of SSHKit are defined as subclasses of
SSHKit::StandardError
which is itself a subclass ofStandardError
.
The Command
objects can be set to not raise, by passing raise_on_non_zero_exit: false
when instantiating them, this is exactly what test()
does internally.
Example:
on hosts do |host
if test "[ -d /opt/sites ]" do
within "/opt/sites" do
execute :git, :pull
end
else
execute :git, :clone, 'some-repository', '/opt/sites'
end
end
- Support arbitrary properties on Host objects. (see below)
Starting with this version, the Host
class supports arbitrary properties,
here's a proposed use-case:
servers = %w{one.example.com two.example.com
three.example.com four.example.com}.collect do |s|
h = SSHKit::Host.new(s)
if s.match /(one|two)/
h.properties.roles = [:web]
else
h.properties.roles = [:app]
end
end
on servers do |host|
if host.properties.roles.include?(:web)
# Do something pertinent to web servers
elsif host.properties.roles.include?(:app)
# Do something pertinent to application servers
end
end
Naturally, this is a contrived example, the #properties
attribute on the
Host instance is implemented as an OpenStruct
and
will behave exactly as such.
- Removed configuration option
SSHKit.config.format
(see below) - Removed configuration option
SSHKit.config.runner
(see below)
The format should now be set by doing:
SSHKit.config.output = File.open('/dev/null')
SSHKit.config.output = MyFormatterClass.new($stdout)
The library ships with three formatters, BlackHole
, Dot
and Pretty
.
The default is Pretty
, but can easily be changed:
SSHKit.config.output = SSHKit::Formatter::Pretty.new($stdout)
SSHKit.config.output = SSHKit::Formatter::Dot.new($stdout)
SSHKit.config.output = SSHKit::Formatter::BlackHole.new($stdout)
The one and only argument to the formatter is the String/StringIOish object to which the output should be sent. (It should be possible to stack formatters, or build a multi-formatter to log, and stream to the screen, for example)
The runner is now set by default_options
on the Coordinator class. The
default is still :parallel, and can be overridden on the on()
(or
Coordinator#each
) calls directly.
There is no global way to change the runner style for all on()
calls as of
version 0.0.5
.
- Rename the ConnectionManager class to Coordinator, connections are handled in the backend, if it needs to create some connections.
- Refactor the runner classes into an abstract heirarchy.
- Include a Pretty formatter
- Modify example to use Pretty formatter.
- Move common behaviour to an abstract formatter.
- Formatters no longer inherit StringIO
First release.