CfScript
CfScript is a DSL for scripting the Cloud Foundry CLI, with a focus on application deployment and management. See Supported Commands
# example.rb
require 'cf_script'
cf space: :development do
  workers = apps ending_with: 'worker'
  workers.each do |worker|
    app worker do
      show :state, :memory, :instances
      stop if started?
      progress 'Setting ENV variable...'
      set_env 'SOME_NAME', 'SOME_VALUE'
      progress "Restaging #{name} in #{current_space}..."
      restage
    end
  end
endTo see the cf commands that get executed, run with TRACE=1.
TRACE=1 ruby example.rb
Requirements
- 
ruby>= 2.2
- 
cf>= 6.0
Install
Install the gem with:
gem install cf_script
Or, add it to your project's Gemfile:
gem 'cf_script'
Execution Scopes
There are three execution scopes/blocks; cf, space, and app:
The cf Block Scope
The cf scope is the top-level scope, it includes evertything between the
do/end following a call to cf. The scope saves the current target on
entry and restores it on exit.
# cf target => staging
cf space: :development do
	# cf target => development
end
# cf target => stagingThe cf block accepts the following options:
- 
api: Sets the API endpoint (optional)
- 
org: Sets the target organization (optional)
- 
username/password: Credentials to use for logging in. (optional)
- 
space: Sets the target space (optional)
Note: To save the current target the cf block will execute cf target
on entry, and if it changed within the block, it wil execute cf target SPACE
on exit to restore it.
The space Block Scope
The space scope is a sub-scope and can only appear within a cf block. It
also saves the current target on entry and restores, if changed, it on exit.
# before cf block, target => staging
cf space: :development do
  # inside cf block, target => development
  space :production do
    # inside space block, target => production
  end
  # still inside cf block, target => development
end
# after cf block, target => stagingThe app Block Scope
The app scope can appear within a cf or space blocks. Within the block
the selected app is the target of commands, this means that commands that take
an application name as their first argument can be called without it.
cf do
  app :api do
    env.each do |name, value|
      unset_env name
    end
    started? ? restart : start
  end
endThe app method accepts a string or an AppInfo object as a name argument.
The apps Method
The apps method mirrors the cf apps command. When called without a block,
it returns an AppList
object that acts as an array of AppInfo
objects, each of which contains the information in each row output by cf apps.
cf do
  # Get the list of AppInfo objects
  space_apps = apps
  # Iterate over the AppInfo objects
  apps.each do |app_info|
    # ...
  end
endWhen the apps method is called with a block, it excutes the block within the
context of each app.
cf do
  apps do
    started? ? restart : start
  end
endThis is equivalent to:
cf do
  apps.each do |app_info|
    app app_info do
      started? ? restart : start
    end
  end
endSupported Commands
| Commands | CLI command | ⋯ | 
|---|---|---|
| General | ||
| api | cf api URL | ✓ | 
| auth | cf auth USER PASSWORD | ✓ | 
| login | cf login -u USER -p PASSWORD [OPTIONS] | ✓ | 
| logout | cf logout | ✓ | 
| target | cf target [-s SPACE -o ORG] | ✓ | 
| Applications | ||
| apps | cf apps | ✓ | 
| app | cf app APP_NAME | ✓ | 
| start | cf start APP_NAME | ✓ | 
| stop | cf stop APP_NAME | ✓ | 
| restart | cf restart APP_NAME | ✓ | 
| push | cf push APP_NAME [OPTIONS] | ✓ | 
| restage | cf restage APP_NAME | ✓ | 
| scale | cf scale APP_NAME [OPTIONS] | ✓ | 
| rename | cf rename APP_NAME NEW_APP_NAME | ✓ | 
| delete | cf delete APP_NAME [-f -r] | ✓ | 
| env | cf env APP_NAME | ✓ | 
| set-env | cf set-env APP_NAME VAR_NAME VAR_VALUE | ✓ | 
| unset-env | cf unset-env APP_NAME VAR_NAME | ✓ | 
| restart-app-instance | cf restart-app-instance APP_NAME INDEX | ✓ | 
| Routes | ||
| routes | cf routes | ✓ | 
| check-route | cf check-route HOST DOMAIN | ✓ | 
| create-route | cf create-route SPACE DOMAIN [-n HOSTNAME] | ✓ | 
| map-route | cf map-route APP_NAME DOMAIN [-n HOSTNAME] | ✓ | 
| unmap-route | cf unmap-route APP_NAME DOMAIN [-n HOSTNAME] | ✓ | 
| delete-route | cf delete-route DOMAIN [-n HOSTNAME] [-f] | ✓ | 
| Spaces | ||
| spaces | cf spaces | ✓ | 
| space | cf space SPACE | ✓ | 
Building
The project uses the standard rubygems package tasks, so:
To build your cloned copy of the gem, run:
rake build
To install the gem from the cloned project, run:
rake install
Known Problems/Limitations
- 
Obviously everything depends on the output of cf, so if that changes things will break. This is not unique to cf_script, but it is something worth noting. The centralized parsing ofcf's output should reduce the impact of change on scripts, but there is no guarantee.
- 
The fixtures were collected manually and they might become a pain to manage. Will investigate creating a tool/script that automates their collection from cf.
- 
I18N, n'existe pas, mais, it's not impossible. If implemented, it should be based on the latest translation files from the cfrepository, and the process of pulling the files automated. In the meantime, might explicity add LANG to the ENV variables passed tocf.
- 
Windows? No clues. Probably not without some changes.