Restforce travis-ci Code Climate

Restforce is a ruby gem for the Salesforce REST api. It's meant to be a lighter weight alternative to the databasedotcom gem.

It attempts to solve a couple of key issues that the databasedotcom gem has been unable to solve:

  • Support for interacting with multiple users from different orgs.
  • Support for parent-to-child relationships.
  • Support for aggregate queries.
  • Remove the need to materialize constants.
  • Support for the Streaming API
  • Support for blob data types.
  • A clean and modular architecture using Faraday middleware

Installation

Add this line to your application's Gemfile:

gem 'restforce'

And then execute:

$ bundle

Or install it yourself as:

$ gem install restforce

Usage

Restforce is designed with flexibility and ease of use in mind. By default, all api calls will return Hashie::Mash objects, so you can do things like client.query('select Id, (select Name from Children__r) from Account').Children__r.first.Name.

Initialization

Which authentication method you use really depends on your use case. If you're building an application where many users from different orgs are authenticated through oauth and you need to interact with data in their org on their behalf, you should use the OAuth token authentication method.

If you're using the gem to interact with a single org (maybe you're building some salesforce integration internally?) then you should use the username/password authentication method.

OAuth token authentication

client = Restforce.new :oauth_token => 'oauth token',
  :instance_url  => 'instance url'

Although the above will work, you'll probably want to take advantage of the (re)authentication middleware by specifying a refresh token, client id and client secret:

client = Restforce.new :oauth_token => 'oauth token',
  :refresh_token => 'refresh token',
  :instance_url  => 'instance url',
  :client_id     => 'client_id',
  :client_secret => 'client_secret'

Username/Password authentication

If you prefer to use a username and password to authenticate:

client = Restforce.new :username => 'foo',
  :password       => 'bar',
  :security_token => 'security token'
  :client_id      => 'client_id',
  :client_secret  => 'client_secret'

Sandbox Orgs

You can connect to sandbox orgs by specifying a host. The default host is 'login.salesforce.com':

client = Restforce.new :host => 'test.salesforce.com'

Global configuration

You can set any of the options passed into Restforce.new globally:

Restforce.configure do |config|
  config.client_id     = ENV['SALESFORCE_CLIENT_ID']
  config.client_secret = ENV['SALESFORCE_CLIENT_SECRET']
end

Query

accounts = client.query("select Id, Something__c from Account where Id = 'someid'")
# => #<Restforce::Collection >

 = records.first
# => #<Restforce::SObject >

.Id
# => "someid"

.Name = 'Foobar'
.save
# => true

.destroy
# => true
# Find all occurrences of 'bar'
client.search('FIND {bar}')
# => #<Restforce::Collection >

# Find accounts match the term 'genepoint' and return the Name field
client.search('FIND {genepoint} RETURNING Account (Name)').map(&:Name)
# => ['GenePoint']

Create

# Add a new account
client.create('Account', Name: 'Foobar Inc.')
# => '0016000000MRatd'

Update

# Update the Account with Id '0016000000MRatd'
client.update('Account', Id: '0016000000MRatd', Name: 'Whizbang Corp')
# => true

Upsert

# Update the record with external ID of 12
client.upsert('Account', 'External__c', External__c: 12, Name: 'Foobar')

Destroy

# Delete the Account with Id '0016000000MRatd'
client.destroy('Account', '0016000000MRatd')
# => true

File Uploads

Using the new Blob Data api feature (500mb limit):

client.create 'Document', FolderId: '00lE0000000FJ6H',
  Description: 'Document test',
  Name: 'My image',
  Body: Restforce::UploadIO.new(File.expand_path('image.jpg', __FILE__), 'image/jpeg'))

Using base64 encoded data (37.5mb limit):

client.create 'Document', FolderId: '00lE0000000FJ6H',
  Description: 'Document test',
  Name: 'My image',
  Body: Base64::encode64(File.read('image.jpg'))

Streaming

Restforce supports the Streaming API, and makes implementing pub/sub with Salesforce a trivial task:

# Initialize a client with your username/password/oauth token/etc
client = Restforce.new

# Force an authentication request
client.authenticate!

EM.run {
  # Assuming you've setup a PushTopic called 'AllAccounts' (See the link above.)
  client.subscribe 'AllAccounts' do |message|
    puts message.inspect
  end
}

Boom, you're now receiving push notifications when Accounts are created/updated.

Caching

The gem supports easy caching of GET requests (e.g. queries):

# Memcached example:

cache = Dalli::Client.new

client = Restforce.new cache: cache

# or

Restforce.configure do |config|
  config.cache = cache
end

Logging/Debugging

You can easily inspect what Restforce is sending/receiving by setting Restforce.log = true.

Restforce.log = true
client = Restforce.new.query('select Id, Name from Account')

# => I, [2012-09-11T21:54:00.488991 #24032]  INFO -- : post https://login.salesforce.com/services/oauth2/token
# => D, [2012-09-11T21:54:00.489078 #24032] DEBUG -- request: 
# => I, [2012-09-11T21:54:00.997295 #24032]  INFO -- Status: 200
# => D, [2012-09-11T21:54:00.997391 #24032] DEBUG -- response headers: server: ""
# => content-type: "application/json; charset=UTF-8"
# => transfer-encoding: "chunked"
# => date: "Wed, 12 Sep 2012 04:53:59 GMT"
# => connection: "close"
# => D, [2012-09-11T21:54:00.997431 #24032] DEBUG -- response body: { ... }
# => I, [2012-09-11T21:54:00.998985 #24032]  INFO -- : get https://na9.salesforce.com/services/data/v24.0/query?q=select+Id%2C+Name+from+Account
# => D, [2012-09-11T21:54:00.999040 #24032] DEBUG -- request: Authorization: "OAuth token"
# => I, [2012-09-11T21:54:01.622874 #24032]  INFO -- Status: 200
# => D, [2012-09-11T21:54:01.623001 #24032] DEBUG -- response headers: server: ""
# => content-type: "application/json; charset=UTF-8"
# => transfer-encoding: "chunked"
# => date: "Wed, 12 Sep 2012 04:54:00 GMT"
# => connection: "close"
# => D, [2012-09-11T21:54:01.623058 #24032] DEBUG -- response body: { ... }

Contributing

  1. Fork it
  2. Create your feature branch (git checkout -b my-new-feature)
  3. Commit your changes (git commit -am 'Added some feature')
  4. Push to the branch (git push origin my-new-feature)
  5. Create new Pull Request