Build Status Coverage Status

NPM

Patio

Patio is a Sequel inspired query engine.

Installation

To install patio run

npm install comb patio

If you want to use the patio executable for migrations

npm install -g patio

Running Tests

To run the tests

  1. make test

To run just the postgres tests

  1. make test-pg

To run just the mysql tests

  1. make test-mysql

Why Use Patio?

Patio is different because it allows the developers to choose the level of abtraction they are comfortable with.

If you want to use the ORM functionality you can. If you don't you can just use the Database and Datasets as a querying API, and if you need to you can write plain SQL

Concepts

  1. Model definitions are defined by the tables in the database.

    As you add models the definition is automatically defined from the table definition. This is particularly useful when you want to define your model from a schema designed using another tool (i.e. ActiveRecord, Sequel, etc...)

  2. Patio tries to stay out of your way when querying.

    When you define a model you still have the freedom to do any type of query you want.

    Only want certain columns?

    1. MyModel.select("id", "name", "created").forEach(function(record){
    2. //record only has the id, name, and created columns
    3. });

    You want to join with another table?

    1. MyModel.join("otherTable", {id: patio.sql.identifier("myModelId"}).forEach(function(record){
    2. //Record has columns from your join table now!
    3. });

    You want to run raw SQL?

    1. MyModel.db.run("select * from my_model where name = 'Bob'").all().chain(function(records){
    2. //all records with a name that equals bob.
    3. });

    You want to just query the database and not use a model?

    1. var DB = patio.connect("pg://test:test@127.0.0.1:5432/test_db");
    2. DB.from("myTable").filter({id: [1,2,3]}).all().function(records){
    3. //records with id IN (1,2,3)
    4. });

Getting Started

All the code for this example can be found here

  1. Create a new database

    PostgreSQL

    1. psql -c "CREATE DATABASE reademe_example"

    MySQL

    1. mysql -e "CREATE DATABASE readme_example"
  2. Create a migration

    1. mkdir migration
    2. patio migration-file -n createInitialTables ./migration

    This will add a migration name createdInitialTables in your migration directory.

  3. Add the following code to your migration

    1. module.exports = {
    2. //up is called when you migrate your database up
    3. up: function (db) {
    4. //create a table called state;
    5. return db
    6. .createTable("state", function () {
    7. this.primaryKey("id");
    8. this.name(String);
    9. this.population("integer");
    10. this.founded(Date);
    11. this.climate(String);
    12. this.description("text");
    13. })
    14. .chain(function () {
    15. //create another table called capital
    16. return db.createTable("capital", function () {
    17. this.primaryKey("id");
    18. this.population("integer");
    19. this.name(String);
    20. this.founded(Date);
    21. this.foreignKey("stateId", "state", {key: "id", onDelete: "CASCADE"});
    22. });
    23. });
    24. },
    25. //down is called when you migrate your database down
    26. down: function (db) {
    27. //drop the state and capital tables
    28. return db.dropTable("capital", "state");
    29. }
    30. };
  4. Run your migration

    1. patio migrate -v --camelize -u "<DB_CONNECTION_STRING>" -d ./migration
  1. Connect and query!

    1. var patio = require("patio");
    2. //set camelize = true if you want snakecase database columns as camelcase
    3. patio.camelize = true;
    4. patio.connect("pg://postgres@127.0.0.1:5432/readme_example");
    5. //define a State model with a relationship to capital
    6. var State = patio.addModel("state").oneToOne("capital");
    7. //define a Capital model with a relationship to State
    8. var Capital = patio.addModel("capital").manyToOne("state");
    9. //save a state
    10. State
    11. .save({
    12. name: "Nebraska",
    13. population: 1796619,
    14. founded: new Date(1867, 2, 4),
    15. climate: "continental",
    16. //notice the capital relationship is inline
    17. capital: {
    18. name: "Lincoln",
    19. founded: new Date(1856, 0, 1),
    20. population: 258379
    21. }
    22. })
    23. .chain(function () {
    24. //save a Capital
    25. return Capital.save({
    26. name: "Austin",
    27. founded: new Date(1835, 0, 1),
    28. population: 790390,
    29. //define the state inline
    30. state: {
    31. name: "Texas",
    32. population: 25674681,
    33. founded: new Date(1845, 11, 29)
    34. }
    35. });
    36. })
    37. .chain(function () {
    38. //Query all the states by name
    39. return State.order("name").forEach(function (state) {
    40. //Get the associated capital
    41. return state.capital.chain(function (capital) {
    42. console.log("%s's capital is %s.", state.name, capital.name);
    43. });
    44. });
    45. })
    46. .chain(process.exit, function (err) {
    47. console.log(err)
    48. process.exit(1);
    49. });

Guides

Features


Documentation generated using coddoc.