aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--railties/CHANGELOG2
-rw-r--r--railties/lib/rails_generator/generators/components/scaffold/scaffold_generator.rb7
-rw-r--r--railties/test/generators/generator_test_helper.rb177
-rw-r--r--railties/test/generators/rails_scaffold_generator_test.rb167
4 files changed, 350 insertions, 3 deletions
diff --git a/railties/CHANGELOG b/railties/CHANGELOG
index 8aecfe1cbf..74dd66f9e7 100644
--- a/railties/CHANGELOG
+++ b/railties/CHANGELOG
@@ -1,5 +1,7 @@
*SVN*
+* Scaffold generator tests. #8443 [pelle]
+
* Generated scaffold functional tests use assert_difference. #8421 [norbert]
* Update to Prototype 1.5.1. [Sam Stephenson]
diff --git a/railties/lib/rails_generator/generators/components/scaffold/scaffold_generator.rb b/railties/lib/rails_generator/generators/components/scaffold/scaffold_generator.rb
index 15eb269d1e..8a331867eb 100644
--- a/railties/lib/rails_generator/generators/components/scaffold/scaffold_generator.rb
+++ b/railties/lib/rails_generator/generators/components/scaffold/scaffold_generator.rb
@@ -5,9 +5,10 @@ class ScaffoldGenerator < Rails::Generator::NamedBase
:controller_class_nesting,
:controller_class_nesting_depth,
:controller_class_name,
+ :controller_underscore_name,
:controller_singular_name,
:controller_plural_name
- alias_method :controller_file_name, :controller_singular_name
+ alias_method :controller_file_name, :controller_underscore_name
alias_method :controller_table_name, :controller_plural_name
def initialize(runtime_args, runtime_options = {})
@@ -16,8 +17,8 @@ class ScaffoldGenerator < Rails::Generator::NamedBase
@controller_name = @name.pluralize
base_name, @controller_class_path, @controller_file_path, @controller_class_nesting, @controller_class_nesting_depth = extract_modules(@controller_name)
- @controller_class_name_without_nesting, @controller_singular_name, @controller_plural_name = inflect_names(base_name)
-
+ @controller_class_name_without_nesting, @controller_underscore_name, @controller_plural_name = inflect_names(base_name)
+ @controller_singular_name=base_name.singularize
if @controller_class_nesting.empty?
@controller_class_name = @controller_class_name_without_nesting
else
diff --git a/railties/test/generators/generator_test_helper.rb b/railties/test/generators/generator_test_helper.rb
new file mode 100644
index 0000000000..0bfb2a0964
--- /dev/null
+++ b/railties/test/generators/generator_test_helper.rb
@@ -0,0 +1,177 @@
+module GeneratorTestHelper
+ # Instatiates the Generator
+ def build_generator(name,params)
+ Rails::Generator::Base.instance(name,params)
+ end
+
+ # Runs the create command (like the command line does)
+ def run_generator(name,params)
+ silence_generator do
+ build_generator(name,params).command(:create).invoke!
+ end
+ end
+
+ # Silences the logger temporarily and returns the output as a String
+ def silence_generator
+ logger_original=Rails::Generator::Base.logger
+ myout=StringIO.new
+ Rails::Generator::Base.logger=Rails::Generator::SimpleLogger.new(myout)
+ yield if block_given?
+ Rails::Generator::Base.logger=logger_original
+ myout.string
+ end
+
+ # asserts that the given controller was generated.
+ # It takes a name or symbol without the <tt>_controller</tt> part and an optional super class.
+ # The contents of the class source file is passed to a block.
+ def assert_generated_controller_for(name,parent="ApplicationController")
+ assert_generated_class "app/controllers/#{name.to_s.underscore}_controller",parent do |body|
+ yield body if block_given?
+ end
+ end
+
+ # asserts that the given model was generated.
+ # It takes a name or symbol and an optional super class.
+ # the contents of the class source file is passed to a block.
+ def assert_generated_model_for(name,parent="ActiveRecord::Base")
+ assert_generated_class "app/models/#{name.to_s.underscore}",parent do |body|
+ yield body if block_given?
+ end
+ end
+
+ # asserts that the given helper was generated.
+ # It takes a name or symbol without the <tt>_helper</tt> part
+ # the contents of the module source file is passed to a block.
+ def assert_generated_helper_for(name)
+ assert_generated_module "app/helpers/#{name.to_s.underscore}_helper" do |body|
+ yield body if block_given?
+ end
+ end
+
+ # asserts that the given functional test was generated.
+ # It takes a name or symbol without the <tt>_controller_test</tt> part and an optional super class.
+ # the contents of the class source file is passed to a block.
+ def assert_generated_functional_test_for(name,parent="Test::Unit::TestCase")
+ assert_generated_class "test/functional/#{name.to_s.underscore}_controller_test",parent do |body|
+ yield body if block_given?
+ end
+ end
+
+ # asserts that the given unit test was generated.
+ # It takes a name or symbol without the <tt>_test</tt> part and an optional super class.
+ # the contents of the class source file is passed to a block.
+ def assert_generated_unit_test_for(name,parent="Test::Unit::TestCase")
+ assert_generated_class "test/unit/#{name.to_s.underscore}_test",parent do |body|
+ yield body if block_given?
+ end
+ end
+
+ # asserts that the given file was generated.
+ # the contents of the file is passed to a block.
+ def assert_generated_file(path)
+ assert_file_exists(path)
+ File.open("#{RAILS_ROOT}/#{path}") do |f|
+ yield f.read if block_given?
+ end
+ end
+
+ # asserts that the given file exists
+ def assert_file_exists(path)
+ assert File.exists?("#{RAILS_ROOT}/#{path}"),"The file '#{path}' should exist"
+ end
+
+ # asserts that the given class source file was generated.
+ # It takes a path without the <tt>.rb</tt> part and an optional super class.
+ # the contents of the class source file is passed to a block.
+ def assert_generated_class(path,parent=nil)
+ path=~/\/?(\d+_)?(\w+)$/
+ class_name=$2.camelize
+ assert_generated_file("#{path}.rb") do |body|
+ assert body=~/class #{class_name}#{parent.nil? ? '':" < #{parent}"}/,"the file '#{path}.rb' should be a class"
+ yield body if block_given?
+ end
+ end
+
+ # asserts that the given module source file was generated.
+ # It takes a path without the <tt>.rb</tt> part.
+ # the contents of the class source file is passed to a block.
+ def assert_generated_module(path)
+ path=~/\/?(\w+)$/
+ module_name=$1.camelize
+ assert_generated_file("#{path}.rb") do |body|
+ assert body=~/module #{module_name}/,"the file '#{path}.rb' should be a module"
+ yield body if block_given?
+ end
+ end
+
+ # asserts that the given css stylesheet file was generated.
+ # It takes a path without the <tt>.css</tt> part.
+ # the contents of the stylesheet source file is passed to a block.
+ def assert_generated_stylesheet(path)
+ assert_generated_file("public/stylesheets/#{path}.css") do |body|
+ yield body if block_given?
+ end
+ end
+
+ # asserts that the given yaml file was generated.
+ # It takes a path without the <tt>.yml</tt> part.
+ # the parsed yaml tree is passed to a block.
+ def assert_generated_yaml(path)
+ assert_generated_file("#{path}.yml") do |body|
+ assert yaml=YAML.load(body)
+ yield yaml if block_given?
+ end
+ end
+
+ # asserts that the given fixtures yaml file was generated.
+ # It takes a fixture name without the <tt>.yml</tt> part.
+ # the parsed yaml tree is passed to a block.
+ def assert_generated_fixtures_for(name)
+ assert_generated_yaml "test/fixtures/#{name.to_s.underscore}" do |yaml|
+ yield yaml if block_given?
+ end
+ end
+
+ # asserts that the given views were generated.
+ # It takes a controller name and a list of views (including extensions).
+ # The body of each view is passed to a block
+ def assert_generated_views_for(name,*actions)
+ actions.each do |action|
+ assert_generated_file("app/views/#{name.to_s.underscore}/#{action.to_s}") do |body|
+ yield body if block_given?
+ end
+ end
+ end
+
+ # asserts that the given migration file was generated.
+ # It takes the name of the migration as a parameter.
+ # the migration body is passed to a block.
+ def assert_generated_migration(name,parent="ActiveRecord::Migration")
+ assert_generated_class "db/migrate/001_#{name.to_s.underscore}",parent do |body|
+ yield body if block_given?
+ end
+ end
+
+ # asserts that the given resource was added to the routes.
+ def assert_added_route_for(name)
+ assert_generated_file("config/routes.rb") do |body|
+ assert body=~/map.resources :#{name.to_s.underscore}/,"should add route for :#{name.to_s.underscore}"
+ end
+ end
+
+ # asserts that the given methods are defined in the body.
+ # This does assume standard rails code conventions with regards to the source code.
+ # The body of each individual method is passed to a block.
+ def assert_has_method(body,*methods)
+ methods.each do |name|
+ assert body=~/^ def #{name.to_s}\n((\n| .*\n)*) end/,"should have method #{name.to_s}"
+ yield( name, $1 ) if block_given?
+ end
+ end
+
+ # asserts that the given column is defined in the migration
+ def assert_generated_column(body,name,type)
+ assert body=~/t\.#{type.to_s} :#{name.to_s}/, "should have column #{name.to_s} defined"
+ end
+
+end \ No newline at end of file
diff --git a/railties/test/generators/rails_scaffold_generator_test.rb b/railties/test/generators/rails_scaffold_generator_test.rb
new file mode 100644
index 0000000000..cbf116996c
--- /dev/null
+++ b/railties/test/generators/rails_scaffold_generator_test.rb
@@ -0,0 +1,167 @@
+require 'test/unit'
+
+# Optionally load RubyGems.
+begin
+ require 'rubygems'
+rescue LoadError
+end
+
+# Mock out what we need from AR::Base.
+module ActiveRecord
+ class Base
+ class << self
+ attr_accessor :pluralize_table_names
+ end
+ self.pluralize_table_names = true
+ end
+
+ module ConnectionAdapters
+ class Column
+ attr_reader :name, :default, :type, :limit, :null, :sql_type, :precision, :scale
+
+ def initialize(name, default, sql_type = nil)
+ @name=name
+ @default=default
+ @type=@sql_type=sql_type
+ end
+
+ def human_name
+ @name.humanize
+ end
+ end
+ end
+end
+
+# And what we need from ActionView
+module ActionView
+ module Helpers
+ module ActiveRecordHelper; end
+ class InstanceTag; end
+ end
+end
+
+
+# Must set before requiring generator libs.
+if defined?(RAILS_ROOT)
+ RAILS_ROOT.replace "#{File.dirname(__FILE__)}/../fixtures/tmp"
+else
+ RAILS_ROOT = "#{File.dirname(__FILE__)}/../fixtures/tmp"
+end
+Dir.mkdir(RAILS_ROOT) unless File.exists?(RAILS_ROOT)
+
+$LOAD_PATH.unshift "#{File.dirname(__FILE__)}/../../lib"
+require 'rails_generator'
+require "#{File.dirname(__FILE__)}/generator_test_helper"
+
+class RailsScaffoldGeneratorTest < Test::Unit::TestCase
+
+ include GeneratorTestHelper
+
+ def setup
+ ActiveRecord::Base.pluralize_table_names = true
+ Dir.mkdir("#{RAILS_ROOT}/app") unless File.exists?("#{RAILS_ROOT}/app")
+ Dir.mkdir("#{RAILS_ROOT}/app/views") unless File.exists?("#{RAILS_ROOT}/app/views")
+ Dir.mkdir("#{RAILS_ROOT}/app/views/layouts") unless File.exists?("#{RAILS_ROOT}/app/views/layouts")
+ Dir.mkdir("#{RAILS_ROOT}/config") unless File.exists?("#{RAILS_ROOT}/config")
+ Dir.mkdir("#{RAILS_ROOT}/db") unless File.exists?("#{RAILS_ROOT}/db")
+ Dir.mkdir("#{RAILS_ROOT}/test") unless File.exists?("#{RAILS_ROOT}/test")
+ Dir.mkdir("#{RAILS_ROOT}/test/fixtures") unless File.exists?("#{RAILS_ROOT}/test/fixtures")
+ Dir.mkdir("#{RAILS_ROOT}/public") unless File.exists?("#{RAILS_ROOT}/public")
+ Dir.mkdir("#{RAILS_ROOT}/public/stylesheets") unless File.exists?("#{RAILS_ROOT}/public/stylesheets")
+ File.open("#{RAILS_ROOT}/config/routes.rb",'w') do |f|
+ f<<"ActionController::Routing::Routes.draw do |map|\n\nend\n"
+ end
+ end
+
+ def teardown
+ FileUtils.rm_rf "#{RAILS_ROOT}/app"
+ FileUtils.rm_rf "#{RAILS_ROOT}/test"
+ FileUtils.rm_rf "#{RAILS_ROOT}/config"
+ FileUtils.rm_rf "#{RAILS_ROOT}/db"
+ FileUtils.rm_rf "#{RAILS_ROOT}/public"
+ end
+
+ def test_scaffolded_names
+ g = Rails::Generator::Base.instance('scaffold', %w(ProductLine))
+ assert_equal "ProductLines", g.controller_name
+ assert_equal "ProductLines", g.controller_class_name
+ assert_equal "ProductLine", g.controller_singular_name
+ assert_equal "product_lines", g.controller_plural_name
+ assert_equal "product_lines", g.controller_file_name
+ assert_equal "product_lines", g.controller_table_name
+ end
+
+ def test_scaffold_generates_resources
+
+ run_generator('scaffold', %w(Product))
+
+ assert_generated_controller_for :products do |f|
+
+ assert_has_method f,:index do |name,m|
+ assert m=~/@products = Product\.find\(:all\)/,"#{name} should query products table"
+ end
+
+ assert_has_method f,:show,:edit,:update,:destroy do |name,m|
+ assert m=~/@product = Product\.find\(params\[:id\]\)/,"#{name.to_s} should query products table"
+ end
+
+ assert_has_method f,:new do |name,m|
+ assert m=~/@product = Product\.new/,"#{name.to_s} should instantiate a product"
+ end
+
+ assert_has_method f,:create do |name,m|
+ assert m=~/@product = Product\.new\(params\[:product\]\)/,"#{name.to_s} should instantiate a product"
+ end
+
+ end
+
+ assert_generated_model_for :product
+ assert_generated_functional_test_for :products
+ assert_generated_unit_test_for :product
+ assert_generated_fixtures_for :products
+ assert_generated_helper_for :products
+ assert_generated_stylesheet :scaffold
+ assert_generated_views_for :products, "index.html.erb","new.html.erb","edit.html.erb","show.html.erb"
+ assert_generated_migration :create_products
+ assert_added_route_for :products
+ end
+
+ def test_scaffold_generates_resources_with_attributes
+ run_generator('scaffold', %w(Product name:string supplier_id:integer created_at:timestamp))
+
+ assert_generated_controller_for :products do |f|
+
+ assert_has_method f,:index do |name,m|
+ assert m=~/@products = Product\.find\(:all\)/,"#{name} should query products table"
+ end
+
+ assert_has_method f,:show,:edit,:update,:destroy do |name,m|
+ assert m=~/@product = Product\.find\(params\[:id\]\)/,"#{name.to_s} should query products table"
+ end
+
+ assert_has_method f,:new do |name,m|
+ assert m=~/@product = Product\.new/,"#{name.to_s} should instantiate a product"
+ end
+
+ assert_has_method f,:create do |name,m|
+ assert m=~/@product = Product\.new\(params\[:product\]\)/,"#{name.to_s} should instantiate a product"
+ end
+
+ end
+
+ assert_generated_model_for :product
+ assert_generated_functional_test_for :products
+ assert_generated_unit_test_for :product
+ assert_generated_fixtures_for :products
+ assert_generated_helper_for :products
+ assert_generated_stylesheet :scaffold
+ assert_generated_views_for :products, "index.html.erb","new.html.erb","edit.html.erb","show.html.erb"
+ assert_generated_migration :create_products do |t|
+ assert_generated_column t, :name, :string
+ assert_generated_column t, :supplier_id, :integer
+ assert_generated_column t, :created_at,:timestamp
+ end
+
+ assert_added_route_for :products
+ end
+end