QuickbaseRecord
QuickbaseRecord is a baller ActiveRecord-style ORM for using the Intuit QuickBase platform as a database for Ruby or Ruby on Rails applications.
Installation
Add this line to your application's Gemfile:
gem 'quickbase_record'
And then execute:
$ bundle
Or install it yourself as:
$ gem install quickbase_record
Usage
Initialize the API Client
QuickbaseRecord is built on top of the Advantage Quickbase gem to make the API calls to QuickBase, so you'll need to configure QuickbaseRecord with your app's realm name and provide a valid username and password. This can be done in a single initializer file with a call to QuickbaseRecords.configure, making sure to keep your credentials safe from the world.
# config/initializers/quickbase_record.rb
QuickbaseRecord.configure do |config|
config.realm = "your_apps_realm_name"
config.username = "valid_username" (or something like ENV["QB_USERNAME"])
config.password = "valid_password" (or something like ENV["QB_PASSWORD"])
config.token = "your_app_token_if_applicable"
end
Include it in your Class
Simply include QuickbaseRecord::Model in your class and use the .define_fields method to supply the table's DBID and a mapping of desired field names => QuickBase FIDs
# app/models/post.rb
class Post
include QuickbaseRecord::Model
define_fields ({
dbid: 'abcde12345'
id: 3,
content: 7,
author: 8
})
# code...
end
IMPORTANT: You must supply a key/value pair for :dbid and :id (QuickbaseRecord uses :id instead of :record_id to look more like standard ActiveRecord models)
What You Get
Classes that include QuickbaseRecord::Model and define their fields will have a handful of class and instance methods for interacting with your QuickBase application similar to ActiveRecord. The goal is to be able to use QuickBase as a database and treat your models the same way you would with a traditional database.
@post = Post.find(1) => <Post: @id=1, @content="Amazing post content", @author: 'Cullen Jett'>
--
<%= form_for @post do |f| %>
# code...
<% end %>
--
@post = Post.where(author: 'Cullen Jett').first
@post.update_attributes(author: 'THE Cullen Jett')
--
etc.
Also included/extended are ActiveModel::Naming, ActiveModel::Callbacks, ActiveModel::Validations, and ActiveModel::Conversion (see ActiveModel docs for details). The biggest benefit here is the availability of .validates on your class to validate attributes and capture invalid attributes with #valid?.
Be aware that validations needing context from the database (i.e. validates_uniqueness_of) are not yet supported and will need to be implemented manually.
Database callbacks (i.e. before_save :create_token!) are not fully functional yet, so stay tuned.
Available Methods
.create(attributes_hash)
- Intantiate and save a new object with the given attributes
- Assigns the returned object it's new ID
Post.create(content: 'Amazing post content', author: 'Cullen Jett')
.find(id)
- Query for a specific QuickBase record by it's ID
- Returns a single object
Post.find(params[:id])
.where(attributes_hash)
- Query QuickBase by any field name defined in your class' field mapping
- Returns an array of objects
Post.where(author: 'Cullen Jett').first - Multiple field_name/value pairs are joined with 'AND'
Post.where(id: 1, author: 'Cullen Jett') # {'3'.EX.'1'}AND{'8'.EX.'Cullen Jett'} - Values in an array are joined with 'OR' ``` Post.where(author: ['Cullen Jett', 'Socrates']) # Jett'OR'8''8'.EX'8'.EX.'Socrates'
Post.where({ [author: 'Cullen Jett', id: 123] }) # Jett'OR'3''3'.EX'3'.EX.'123'
- To use a comparator other than 'EX' pass the value as another hash with the key as the comparatorPost.where(author: 'Cullen Jett') # Jett'
- Combine arrays and hashes to build more complex queriesPost.where(id: [123, 'today']) # "'3''3'.XEX'3'.XEX.'123'OR'3''3'.OAF'3'.OAF.'today'"
Post.where(id: 123, OAF: 'today') # "'3''3'.XEX'3'.XEX.'123'AND'3''3'.OAF'3'.OAF.'today'"
- Also accepts a string in the standard QuickBase query format * Works with field names or FIDsPost.query("'3''3'.EX'3'.EX.'1'") Post.query("Jett'")
</code>.qid(id)
- Accepts a QID (QuickBase report ID)
- Returns an array of objects
Post.qid(1)
#save
- Creates a new record in QuickBase for objects that don't have an ID or edits the corresponding QuickBase record if the object already has an ID
- Returns the object (if #save created a record in QuickBase the the returned object will now have an ID)
- Uses API_ImportFromCSV under the hood. ``` @post = Post.new(content: 'Amazing post content', author: 'Cullen Jett') @post.save # => <Post: @id: 1, @content: 'Amazing post content', @author: 'Cullen Jett'
@post.author = 'Socrates' @post.save # => <Post: @id: 1, @content: 'Amazing post content', @author: 'Socrates'
</code>#delete
- Delete the corresponding record in QuickBase
- It returns the object's ID if successful or
falseif unsuccessful@post = Post.find(1) @post.delete
#update_attributes(attributes_hash)
- IMPORTANT: Updates and saves the object with the new attributes
- Returns the object
@post = Post.where(author: 'Cullen Jett').first @post.update_attributes(author: 'Socrates', content: 'Something enlightening...') # => <Post: @id: 1, @author: 'Socrates', @content: 'Something enlightening...'
#assign_attributes(attributes_hash)
- Only changes the objects attributes in memory (i.e. does not save to QuickBase)
- Useful for assigning multiple attributes at once, otherwise you could use the field name's attr_accessor to change a single attribute.
- Returns the object
@post = Post.where(author: 'Cullen Jett').first @post.assign_attributes(author: 'Socrates', content: 'Something enlightening...') @post.save
.qb_client and #qb_client
- Access the quickbase API client (advantage_quickbase gem) directly
File Attachments
When creating an object with a field of type 'file attachment', you must assign it as hash with :name and :file as keys. After the object is saved that field will then become a new hash with :filename and :url as keys.
@post = Post.new(attachment: {name: 'Test File Name', file: 'path/to/your/file OR file contents'})
@post.save
@post. => {filename: 'Test File Name', url: 'https://realm.quickbase.com/up/abcdefg/Test%20File%20Name'}
Testing
Unfortunately you will not be able to run the test suite unless you have access to the QuickBase application used as the test database or you create your own QuickBase app to test against that mimics the test fakes. Eventually the test calls will be stubbed out so anyone can test it, but I've got stuff to do -- pull requests are welcome :)
As of now the tests serve more as documentation for those who don't have access to the testing QuickBase app.
If you're lucky enough to work with me then I can grant you access to the app and you can run the suite until your fingers bleed. You'll just need to modify spec/quickbase_record_config.rb to use your own credentials.
Contributing
- Fork it ( https://github.com/[my-github-username]/quickbase_record/fork )
- 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 a new Pull Request