Skip to content

Latest commit

 

History

History
576 lines (444 loc) · 22.4 KB

README.md

File metadata and controls

576 lines (444 loc) · 22.4 KB

builder

Rake driven application builder from a database schema.

This is an experiment in creating an application generator.

Keep up to date with git pull.

A set of rake tasks and ruby classes to generate various types of application artifacts using a database schema as the source of data. This is useful if you want to create an Amber, Ruby On Rails (with AngularJS) or NodeJS (with Express, MongoDB and Angular) application and you already have a legacy database or you prefer to create the database DDL manually to take advantage of the powerful features of your database engine.

Writing a web application often means writing lots of boilerplate code. The Amber and Rails scaffold generators can be used to generate much of the boilerplate code which is very useful, especially for the views which would otherwise involve lots of tedious keystrokes. The only problem with the rails scaffold generator is that you need to know your database schema up front and then type lots of column_name:column_type arguments to the generator to create the database migrations and the views with elements for all the fields.

This project does much the same thing only the emphasis is on generating application artifacts using a database schema as the source of information about columns and their types and it also generates artifacts for an AngularJS Single Page Application.

Note: Because the frameworks change often but the scaffold generator arguments change rarely, the best way to use this tool is to generate a set of commands to generate artifacts using the framework's scaffold generator. Consequently the only generators that are now recommended are the Amber and RailsVanilla generators.

Amber: Amber is a web application framework for the Crystal language programming language. Crystal syntax is heavily inspired by Ruby and Amber shares many conventions with Rails so Ruby on Rails developers can get up and running with Amber very quickly. This project will take you go one step further by helping you to get an Amber project up and running for an existing database with the default CRUD operations. Amber also has application recipes, so you can generate applications with many different layouts and SPA frontend frameworks, you aren't limited to the default internal generator.

Once you have generated the artifacts for your application you can then build your app using the generated artifacts as a base. A lot of the donkey work has been done for you.

You can utilize a couple of things to keep generated artifacts separate from modified/static artifacts.

Use the gem rack-static_fallback to load a resource from a fallback location. Add the gem to your Gemfile and add the following to your config/environments/[development|production].rb

config.middleware.use ::Rack::StaticFallback,
  :mode => :fallback,
  :static_path_regex => %r{/(views|partials)},
  :fallback_static_url => "/generated/"

For a given url of '/views/user/form.html', if the resource form.html exists in 'public/views/user' then use that otherwise try 'public/generated/views/user/form.html'. Thus views and partials are generated in public/generated and if you want to modify one of the generated artifacts then first copy it to the corresponding path in public/views or public/partials.

In app/assets/javascripts/application.js override the generated artifacts with modified/artifacts. E.g.

//= require app/app
//= require app/routes
//= require app/services
//= require app/controllers
//= require app/directives
// include the generated modules first
//= require_tree ./generated
// override the generated modules with static modules
//= require_tree ./modules

Dependencies

  • Mysql2 or JDBC-Mysql connector
  • Rake

Additional dependencies

If you are not using the Amber builder then additional dependencies may be required

  • Erubis
  • AngularUI (for datepicker)
  • Restangular (and underscore or lodash) if using RestAngularBuilder

Installation

git clone https://github.com/damianham/builder.git

Usage

Step 1 - generate the column info schema file from the database

 rake -f builder/tasks/mysql2.rake build_schema[dbhostname,testdb,testuser,testpass]    

Generates a testdb_column_info.yml file from the testdb database schema on host using a username of testuser and password of testpass using the mysql2 gem. Use jdbc_mysql.rake if using JRuby. To get the best results the tables in the database should define foreign key constraints and use table and column comments throughout. For example for MySQL we can create a set of related tables with

create table `users` (
  `id` int(10) NOT NULL AUTO_INCREMENT PRIMARY KEY,
  `username` varchar(32) NOT NULL COMMENT 'The user identity used to login to the system',
  `password` varchar(64) NOT NULL COMMENT 'MD5 digest of the users password',
  `secret` varchar(255) NOT NULL COMMENT 'The users answer to the secret question for account recovery', 
  `updated_at` TIMESTAMP COMMENT 'Rails convention - date time the record was last updated',
  `created_at` DATETIME COMMENT 'Rails convention - date time the record was initially created'
) COMMENT 'Users are people who can login to the system and create interesting posts and witty comments';

create table `posts` (
  `id` int(10) NOT NULL AUTO_INCREMENT PRIMARY KEY,
  `title` varchar(255) NOT NULL COMMENT 'The title of the post that will appear in lists, tables and headings',
  `content` text NOT NULL COMMENT 'The main body content of the post',
  `user_id` int(10) NOT NULL COMMENT 'Foreign key to the user record in the users table that created the post',
  `updated_at` TIMESTAMP COMMENT 'Rails convention - date time the record was last updated',
  `created_at` DATETIME COMMENT 'Rails convention - date time the record was initially created',
  KEY `posts_user_id` (`user_id`),
  CONSTRAINT `posts_user_id_fk` FOREIGN KEY (`user_id`) REFERENCES `users`(`id`)
) COMMENT 'Posts are created by users and contain really interesting information';

create table `comments` ( 
  `id` int(10) NOT NULL AUTO_INCREMENT PRIMARY KEY,
  `content` text NOT NULL COMMENT 'The main body content of the comment',
  `post_id` int(10) NOT NULL COMMENT 'Foreign key to the post record in the posts table the comment relates to',
  `user_id` int(10) NOT NULL COMMENT 'Foreign key to the user record in the users table that created the comment',
  `updated_at` TIMESTAMP COMMENT 'Rails convention - date time the record was last updated',
  `created_at` DATETIME COMMENT 'Rails convention - date time the record was initially created',
  KEY `comments_post_id` (`post_id`),
  KEY `comments_user_id` (`user_id`),
  CONSTRAINT `comments_post_id_fk` FOREIGN KEY (`post_id`) REFERENCES `posts`(`id`)
  CONSTRAINT `comments_user_id_fk` FOREIGN KEY (`user_id`) REFERENCES `users`(`id`)
) COMMENT 'Comments completely trash the contents of posts - or maybe not';

Step 2 - create any other application structures using other generators

For example you may want to create an Amber or Ruby on Rails application using an application generator

$ amber new mywebapp -d mysql 

# or

$ rails new mywebapp

In particular - with Amber you can use a recipe to generate an application according the structure and templates of the recipe. E.g.

$ amber new -d mysql -r damianham/amber_material_kit

See the Amber documentation for more information on getting started with Amber.

Step 3 - Configure the builder

Copy builder_config.rb to the current working directory and edit the copied file to define the generators and their output folders.

The builders are defined in a hash with the name of the builder as the key and the builder options, including the output folder, as the value. The default builders are

  • AmberBuilder
  • RailsVanillaBuilder

Other generators available are

  • RailsBuilder
  • AngularRailsBuilder
  • RSpecBuilder
  • RestAngularBuilder (use instead of AngularRailsBuilder)
  • RestAngularModalBuilder (use instead of RestAngularBuilder)
  • UIrouterBuilder (uses angular/ui-router - use instead of any other angular builder)
  • MEAN (MongoDB, Express, AngularJS, NodeJS - see http://meanjs.org)

The RailsBuilder generates controllers,models and views and the AngularRailsBuilder generates Angular service factories, controllers and view partials.
If you want to use the scaffold generator that ships with Rails then change RailsBuilder to RailsVanillaBuilder however note that the models generated by the Rails scaffold generator will not have the has_many and belongs_to method calls to define the class relationships, nor will it have the validation method calls for required columns.

For example if you created a new rails application using the rails application generator as described above and you have a directory structure like

workspace

builder

mywebapp

builder_config.rb

then you would set the output folders to the following values

#### define the output destinations relative to where you run the rake command

@builders = {
  RailsBuilder => {:output => './mywebapp'  },
  AngularRailsBuilder => {:output => './mywebapp' }
}

and run the rake commands in the workspace directory with the builder_config.rb file in the workspace directory.

You can also add :only and :except elements to each builder's options hash which are both arrays of method name symbols to limit the actions of each builder.

If the :only option exists then only methods named in the :only array will do anything, all other methods will return immediately. Any methods named in the :except array will do nothing and return immediately.

You can also add the appname option e.g. :appname => 'myapp' for the Angular module name. This is the name of the Angular application and is given as the ng-app name in the html start tag. The default Angular module name is 'mainapp'.

Our example config file now looks like this.

#### define the output destinations relative to where you run the rake command

@builders = {
  RailsBuilder => {:output => './myapp',:appname => 'myapp' ,
  # do not do anything for these methods
  :except => [:finalize_menu ] },

  AngularRailsBuilder => {:output => './myapp',:appname => 'myapp',
  # don't generate the 'app/assets/javascripts/generated/app/app.js' artifact
  :except => [:finalize_application]}
}

It is quite probable that you want to generate an Angular front end that uses a backend api service for data that is namespaced e.g.

/api_v1/users/

You can give an api_prefix option to the Angular builders rather than a namespace in order to access the api at the given path.

Our example config file now looks like this.

#### define the output destinations relative to where you run the rake command

@builders = {
  RailsBuilder => {:output => './myapp',:appname => 'myapp' ,
  :namespace => 'api_v1', 
  :except => [:finalize_menu] },

  AngularRailsBuilder => {:output => './myapp',:appname => 'myapp',
  :api_prefix => 'api_v1',
  :except => [:finalize_application]}
}

define the fields to use for each kind of template for each table

When generating the detail.html, form.html and list.html files for each table/module limit the fields that are generated to the given list.
RSpecBuilder also uses these lists to generate attributes for testing views.
Any column name listed in :except is removed from all tables.

@fields = {
  :except => [:id,:created_at,:updated_at],
  :detail => {
    :users => [:email,:name,:gender,:birthday]  
  },
  :form => {
    :users => [:email,:name,:gender,:birthday]
  },
  :list => {
    :users => [:email,:name,:gender,:birthday],
    :places => [:name, :lat,:long,:map_link]
    }
}

define any tables to ignore

IGNORE_TABLES = ['dodgy_table1','dummy_table2']

Do not generate artifacts for dodgy_table1 and dummy_table2. The Rails migration version table schema_migrations is ignored by the column info generator tasks.

define the type of list template to use - either 'list', or 'table' for index routes

# define the template type for all index routes e.g. /users
# define to 'list' to use the list template
# or 'table' to use the table template
  LIST_TYPE = 'table'

Uses the public/generated/views/{model_name}/{prefix}_{LIST_TYPE}.html template for all index routes. For example, for the route '/contacts' (the model_name is 'contact') and a LIST_TYPE 'table', the prefix used depends on the builder, e.g.

  • AngularRailsBuilder and RestAngularBuilder - prefix == 'partial' giving 'public/generated/views/contact/partial_table.html
  • RestangularModalBuilder - prefix == 'modal' giving 'public/generated/views/contact/modal_table.html
  • UIrouterBuilder - prefix is 'uirouter' giving 'public/generated/views/contact/uirouter_table.html

Along with rack_static_fallback this all comes together in the generated module files in app/assets/javascripts/generated/modules. For example the AngularRailsBuilder and RestAngularBuilder will add the following routes to the angular route provider for the users table for a LIST_TYPE of 'table'.

app/assets/javascripts/generated/modules/UserModule.js

 angular.module('mainapp').config(['$routeProvider',   function (provider) {
        provider
.when('/users/new', {templateUrl:  '/views/user/partial_form.html' ,controller: 'UserFormCtrl'})
.when('/users/:userId/edit', {templateUrl:  '/views/user/partial_form.html' ,controller: 'UserFormCtrl'})
.when('/users/:userId', {templateUrl:  '/views/user/partial_detail.html' ,controller: 'UserDetailCtrl'})
.when('/users', {templateUrl:  '/views/user/partial_table.html' ,controller: 'UserListCtrl'})
 ;
}]);

Step 4 - customize the view templates in the templates folder to your liking

Edit builder/templates/(ng|rails|restangular|mean) .erb template files. These templates are used to generate the artifacts. Note that if you are using the AngularRails builder then the templates that are used are in the folder templates/ng. If you use a RestAngular based builder then some templates are in the folder templates/restangular.

Some form fields need special rendering. For example a date field could be rendered with a date picker rather than as a text field. The AngularUI team provide a set of Bootstrap components written in pure AngularJS that includes a date picker.

You can download the ui.bootstrap library from http://angular-ui.github.io/bootstrap/

For a given model,name,column type you can create field partials that will be used to render the field with the following order of priority

  • model_name.column_name
  • model_name.column_type
  • column_name
  • column_type

so for a column name of birthday with a field type of date in the user model e.g.

create table users (
  birthday date
);

The Angular partial-form view will look for a partial to render for the field in the following order

  • templates/ng/field_partials/models/user/_birthday.erb
  • templates/ng/field_partials/models/user/_date.erb
  • templates/ng/field_partials/by_name/_birthday.erb
  • templates/ng/field_partials/by_type/_date.erb

If none of these files exist then the file

templates/ng/field_partials/_any_field.erb

is used to render the field.

At the moment this only works for Angular Form views.

Step 5 - generate the application artifacts

rake -f builder/tasks/builder.rake build_classes[testdb[,namespace]]

For the Amber and RailsVanilla builders outputs the scaffold generator commands for the framework. Pipe the output through a bash shell in the application folder to generate the scaffold artifacts.

For other builders generates the application artifacts in the output folder using the generated testdb_column_info.yml file as input. The namespace argument will place the artifacts in a subfolder with that namespace. If no appname is supplied in the builder options then the Angular module name will adopt the namespace argument. You can also define a different namespace for each different builder by adding the namespace to the builder options. E.g.

@builders = {
  RailsBuilder => {:output => './myapp',:appname => 'myapp' ,
  :namespace => 'api',  # place the rails artifacts in the 'api' namespace
  # do not do anything for these methods
  :except => [:finalize_menu, :build_model, :build_controller] },

  AngularRailsBuilder => {:output => './myapp',:appname => 'myapp',
  # don't do anything except finalize the menu
  :only => [:build_model, :build_controller]}
}

The footer,header,home,menu partials from builder/templates/ng/footer.html.erb etc. will be generated in mywebapp/public/generated/partials.

The main angular module app.js will be generated in mywebapp/app/assets/javascripts/generated/app along with the services.js module.

An Angular module with a service factory and controllers for the detail,form and list views will be generated from builder/templates/ng/module.js.erb in mywebapp/app/assets/javascripts/generated/modules for each database table.
Routes to the standard CRUD actions (list,detail,new,show) will be added to the module.

The usual Angular way is to generate a controllers in controller.js, services in services.js etc. This approach collects related elements together so the User service factory is in the same place as the UserList and UserForm controller etc.

Html view files for the detail,form and list views for each database table will be generated from builder/templates/ng/partial-detail.erb etc. in mywebapp/public/generated/views/table_name/.

Either builder/templates/ng/partial-table.erb or builder/templates/ng/partial-list.erb will be used to generate the list.html file depending on the value of LIST_TYPE in builder/lib/angular_rails.rb

You can also pass the location of the config file and column info file as environment variables to the rake task

rake -f builder/tasks/builder.rake conf='path_to_config_file' columns='path_to_column_info_file'  build_classes

in which case there is no need to pass the database name as an argument.
Note that if you want to pass a namespace argument then you must also provide the database name first as in build_classes[testdb,api] where api is the namespace.

Step 6 - integrate angular into your rails web application

Integrate the Angular app into your web application by adding the ng-app attribute to the html opening tag of mywebapp/app/views/layouts/application.html.erb using either the appname option from the AngularRailsBuilder config, the namespace option to build_classes or the default 'mainapp'.

<html lang="en" ng-app='mywebapp'>
<head>
        
        <% 
        # add angular-touch.min.js for mobiles
        # load jquery from googleapis CDN - also remove it from application.js
        { "https://ajax.googleapis.com/ajax/libs/jquery/2.1.3/" => [ "jquery.min.js"],
          "http://ajax.googleapis.com/ajax/libs/angularjs/1.3.7/" => 
            ['angular.min.js','angular-animate.min.js','angular-cookies.min.js',
          'angular-resource.min.js','angular-cookies.min.js','angular-loader.min.js',
          'angular-messages.min.js',
          'angular-sanitize.min.js','angular-route.min.js'],
          "http://cdnjs.cloudflare.com/ajax/libs/restangular/1.3.1/" => ['restangular.min.js'],
          "http://cdn.jsdelivr.net/underscorejs/1.7.0/" => ["underscore-min.js"],
          "http://netdna.bootstrapcdn.com/bootstrap/3.2.0/js/" => ["bootstrap.min.js"]
          }.each_pair do |cdn,list|
            list.each do |lib| %>
              <% if Rails.env.production?  %>
                <script src="<%= cdn + lib %>"></script>
              <% else %>
                <script src="/javascript/<%= lib %>"></script> 
              <% end %>
            <% end 
          end
        %>

        <%= stylesheet_link_tag    'application', media: 'all', 'data-turbolinks-track' => true %>
        <%= javascript_include_tag 'application', 'data-turbolinks-track' => true %>
        <%= csrf_meta_tags %>

</head>
<body>
    <!-- the angular view -->
    <div ng-view></div>
</body>
</html>

Add the angular main module and the modules folder to app/assets/javascrips/application.js

//= require jquery
//= require jquery_ujs
//= require turbolinks 

//= require app/app
//= require app/services
//= require_tree ./modules

Step 7 - integrate generated artifacts into your MEANjs web application

If you are using the mean generator integration is simple. Follow the instructions on http://meanjs.org/generator.html to create a new MEAN application. Essentially once you have NodeJS (and thus npm) installed the commands are;

  • npm install -g yo
  • npm install -g generator-meanjs
  • yo meanjs
  • grunt

This should give you a basic working application running on http://localhost:3000

Before running the builder task you may want to add users to the list of ignored tables in the builder_config file as the MEAN application already has useful support for users with authentication from a variety of oauth providers.

Run the builder task and your database tables are added to the MEAN application.

Complete run through

The following steps will create application artifacts from scratch based on an existing mysql database called testdb and a database user with the logon username:password credentials of testuser:testpass

Amber

cd ~/workspace  
git clone https://github.com/damianham/builder.git
rake -f builder/tasks/mysql2.rake build_schema[dbhostname,testdb,testuser,testpass]
cp builder/builder_config.rb .
edit builder_config.rb
amber new mywebapp
rake -f builder/tasks/builder.rake build_classes[testdb] | (cd mywebapp && bash)
cd mywebapp
amber watch

Rails

cd ~/workspace  
git clone https://github.com/damianham/builder.git
rake -f builder/tasks/mysql2.rake build_schema[dbhostname,testdb,testuser,testpass]
rails new mywebapp
cp builder/builder_config.rb .
edit builder_config.rb
rake -f builder/tasks/builder.rake build_classes[testdb]
edit mywebapp/app/views/layouts/application.html.erb
edit mywebapp/app/assets/javascripts/application.js

MEAN

cd ~/workspace  
git clone https://github.com/damianham/builder.git
rake -f builder/tasks/mysql2.rake build_schema[dbhostname,testdb,testuser,testpass]
cp builder/builder_config.rb .
edit builder_config.rb
npm install -g yo
npm install -g generator-meanjs
mkdir mywebapp
(cd mywebapp ; yo meanjs)
rake -f builder/tasks/builder.rake build_classes[testdb]
cd mywebapp
grunt

TODO

  • So many things
  • Create Cucumber, CoffeeScript generators
  • Create schema extractors for other database engines

please fork and contribute