SauceWhisk provides an "ActiveRecord" style client for the Sauce Labs RESTful API. If you're not using the Sauce Gem and you want a nice way to interact with our API, give this a try.
Add this line to your application's Gemfile:
gem 'sauce_whisk', 'Some Version Here'
And then execute:
$ bundle
Or install it yourself as:
$ gem install sauce_whisk
We recommend setting a hard version for now, as the gem is still kinda beta-y.
You'll need a Sauce Labs account. They're free to try and, if you're an open source project, your access is always free.
Value | ENV Variable | Meaning |
---|---|---|
:username | SAUCE_USERNAME | Your Sauce Labs Username |
:access_key | SAUCE_ACCESS_KEY | Your Access Key, found on the lower left of your Account page (Not your password!) |
:asset_fetch_retries | SAUCE_ASSET_FETCH_RETRY | Number of times to retry fetching assets |
:rest_retries | SAUCE_REST_RETRIES | Number of times to try failing REST calls |
There are four ways to configure SauceWhisk. The gem tries each of the following locations in turn. Note, Environment Variables are the preferred means of setting configuration.
username
, access_key
, asset_fetch_retries
and rest_retries
can be set directly on the SauceWhisk object:
SauceWhisk.username = 'some_rad_tester'
SauceWhisk.asset_fetch_retries = 5
If you have the Sauce gem required, the SauceWhisk gem will try to read its configuration from the Sauce gem's configuration.
If you have an ondemand.yml, the SauceWhisk gem will try to read its configuration from that file. The locations the gem looks for a ondemand.yml are, in order:
./ondemand.yml
(The current directory)./config/ondemand.yml
(The config directory under the current directory)$GEM_LOCATION/../ondemand.yml
(The directory of the file which includessauce_whisk
~/.sauce/ondemand.yml
(The .sauce directory in the user's home directory)
The keys are the same as those in the 'value' table, above.
This is the preferred way. Environment variable are available to all applications that need them, won't get accidentally checked into a repo and don't need to be edited for each team member.
SAUCE_USERNAME=Your Sauce Username
SAUCE_ACCESS_KEY=Your Access Key, found on the lower left of your Account page
To configure a proxy, set the 'HTTP_PROXY' or 'http_proxy' environment variable. Proxies need to be a valid URI; HTTP Authentication can be included.
SauceWhisk::Jobs.pass_job job_id
SauceWhisk::Jobs.fail_job job_id
SauceWhisk::Jobs.change_status job_id, true_for_passed_false_for_failed
There are three ways to create a Job object.
# Create an 'empty' job (i.e. don't retrieve job details from the API)
empty_job = SauceWhisk::Job.new job_id
# Create a job with details fetched from the API
fully_detailed_job = SauceWhisk::Jobs.fetch job_id
# Create a job with details fetched from the API, failing if the job is incomplete
partially_detailed_job = SauceWhisk::Jobs.fetch! job_id
Use the first form when you just want a simple way to push details to the API. Use the second form when you want to fetch details from the API, but aren't overly fussy about having complete job information. Use the third form to throw a SauceWhisk::JobNotComplete
exception when fetching an incomplete job. Jobs that are fetched incomplete will not have screenshots, and may not have videos or logs.
NB: It's not possible to create a new job on Sauce Labs' infrastructure with the API.
See here
job = SauceWhisk::Jobs.fetch job_id
job.name = "Determine if the User can Invite Friends"
job.save
job = SauceWhisk::Jobs.fetch job_id
job.build = "12.3.04-beta"
job.visibility = "public"
job.tags = "new_user"
job.name = "Determine if the User can Invite Friends"
job.custom_data = {:executor => "jparth", :team_city_config => "standard_with_instrumentation"}
job.passed = false
job.save
It is not possible to alter any other job properties.
job = SauceWhisk::Jobs.fetch job_id
job.id # Sauce job_id
job.owner # Account name of job's owner
job.browser # Browser job ran on
job.browser_verion # Version of Browser
job.os # OS job ran on
job.log_url # URL for accessing Job Log
job.creation_time # When the job was first created
job.start_time # When the job started running
job.end_time # When the job finished
job.video_url # URL for accessing screencast video
job.screenshot_urls # Array of URLs for accessing screenshots
job.has_all_asset_names? # True if all asset names were fetched (eg job was complete)
Jobs.delete_job job_id
There are three types of asset for Sauce Labs jobs: screenshots, video and logs. Assets are represented as an Asset object, which include the name, asset type and data.
job = SauceWhisk::Jobs.fetch job_id
screenshots = job.screenshots # An array of Screenshot assets
job = SauceWhisk::Jobs.fetch job_id
video = job.video # A single Asset, holding the video
SauceWhisk::Assets.delete job_id #==> Deletes all assets for job.
At the moment, it is only possible to query existing accounts, and create subaccounts.
my_account = SauceWhisk::Accounts.fetch "account_name" # Returns a SauceWhisk::Account object, with its concurrency limit
my_account = SauceWhisk::Accounts.fetch("account_name", false) # Returns a SauceWhisk::Account object, does not fetch its concurrency limit
SauceWhisk::Account
objects store data about the relevant account:
my_account.username # Sauce Labs Username
my_account.access_key # Sauce Labs Access Key (For automated test authentication... Not your password)
my_account.minutes # Automated minutes available for Windows, Android, Linux
my_account.manual_minutes # Manual minutes available for Windows, Android, Linux
my_account.mac_minutes # Automated minutes available for Mac, iOS
my_account.mac_manual_minutes # Manual minutes available for Mac, iOS
my_account.total_concurrency # Number of concurrent jobs allowed
my_account.mac_concurrency # Number of concurrent Mac hosted jobs allowed (includes iPad, iPhone, Mac)
NB: Minutes counts can be strings
If you exceed your concurrency limits, your tests will be queued waiting for a free VM. This may cause erroneous failures in your test cases, and no-one wants that.
concurrencies = SauceWhisk::Accounts.concurrency_for "account_name" # Hash containing both concurrency limits
mac_concurrency = SauceWhisk::Accounts.concurrency_for "account_name", :mac # Fixnum of just the Mac limit
total_concurrency = SauceWhisk::Accounts.concurrency_for "account_name", :total # Fixnum of the upper concurrency limit
parent = SauceWhisk::Accounts.fetch "account_name"
subaccount = parent.add_subaccount("User", "Username", "User@email.com", "Password") # New SauceWhisk::SubAccount
SubAccounts are a subclass of Account, with an accessor for their parent object.
subaccount.parent == parent #=>true
Tunnels give information about currently running Sauce Connect tunnels for a given user.
all_tunnels = SauceWhisk::Tunnels.all # An array of SauceWhisk::Tunnel objects
all_tunnels_as_json = SauceWhisk::Tunnels.all(:fetch_each => false) # An array of Tunnel IDs
tunnel = SauceWhisk::Tunnels.fetch(tunnel_id) # An instance of Tunnel
tunnel.id # The Tunnel, um, id.
tunnel.owner # The Sauce Account responsible for the tunnel -- The Credentials used to open it
tunnel.status # Whether or not the tunnel is open
tunnel.host # The Sauce Labs machine hosting the other end of the Tunnel
tunnel.creation_time # When the tunnel was opened, in Epoch time
tunnel = SauceWhisk::Tunnels.fetch tunnel_id
tunnel.stop # Stops the Tunnel
SauceWhisk::Tunnels.stop tunnel_id # Stops the tunnel with the given id
Occasionally, assets won't be available immediately after the job is finished. The gem will automatically retry fetching them.
You can configure how many times the gem retries by configuring 'asset_fetch_retries` to your desired number of retries.
The Sauce class returns non-user-specific info about available platforms, the number of jobs run, and the status of Sauce Labs' infrastructure.
These can be used without authentication.
You can obtain an Array of all available webdriver platforms using SauceWhisk::Sauce.platforms
:
platforms = SauceWhisk::Sauce.platforms # Fetch all platforms or return cached values
platforms = SauceWhisk::Sauce.platforms(true) # Force a fetch of all platforms
platforms.first # => A Hash of platform details:
{
"long_name"=>"Firefox", # Full name of the platform's browser
"api_name"=>"firefox", # desired_capabilities name of the platform
"long_version"=>"20.0.1.", # Full version number for the platform's browser
"preferred_version"=>"", # Preferred version of the platform's browser (If none is requested)
"automation_backend"=>"webdriver", # Whether this is a Webdriver or Selenium-rc driven platform
"os"=>"Linux", # desired_capabilities name of the Platform's Operating System
"short_version"=>"20" # desired_capabilities name of the Platform's Browsers's version
}
The most important values for a platform are the os, api_name, short_version fields. These are the values to use for desired_capabilites os, browser and browser_version respectively.
The gem does not support retrieval of selenium-rc platforms at the current time
all_tests = SauceWhisk::Sauce.total_job_count # The total number of jobs ever run with Sauce Labs' help
status = SauceWhisk::Sauce.service_status # Check the status of Sauce Labs' service
status.inspect # =>
{
:wait_time=>0.39880952380952384, # Delay between requesting a job and it starting
:service_operational=>true, # Operational Status -- Boolean
:status_message=>"Basic service status checks passed." # More info when erros occuring
}
You can set a custom logger object, passing anything that responds to puts:
SauceWhisk.logger = my_logger
SauceWhisk.logger defaults to STDOUT, at the warn level.
Create a new storage object:
storage = SauceWhisk::Storage.new username: 'my_user_name', key: '00', debug: true
If the environment variables SAUCE_USERNAME and SAUCE_ACCESS_KEY are set then:
storage = SauceWhisk::Storage.new debug: true
List all files in storage.
storage.files
Upload a file.
storage.upload '/tmp/sauce/test.zip'
- Fork the sauce-labs version of this repository
- Create your feature branch (
git checkout -b my-new-feature
) - Commit your changes (
git commit -am 'Add some feature'
) - Push to the branch (
git push origin my-new-feature
) - Create new Pull Request