foca-integrity 0.1.8 → 0.1.9.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (75) hide show
  1. data/README.markdown +7 -0
  2. data/Rakefile +89 -81
  3. data/config/config.sample.ru +11 -21
  4. data/config/config.sample.yml +15 -12
  5. data/lib/integrity.rb +21 -23
  6. data/lib/integrity/app.rb +138 -0
  7. data/lib/integrity/author.rb +39 -0
  8. data/lib/integrity/build.rb +54 -31
  9. data/lib/integrity/commit.rb +71 -0
  10. data/lib/integrity/helpers.rb +3 -3
  11. data/lib/integrity/helpers/authorization.rb +2 -2
  12. data/lib/integrity/helpers/forms.rb +3 -3
  13. data/lib/integrity/helpers/pretty_output.rb +1 -1
  14. data/lib/integrity/helpers/rendering.rb +6 -1
  15. data/lib/integrity/helpers/resources.rb +9 -3
  16. data/lib/integrity/helpers/urls.rb +15 -13
  17. data/lib/integrity/installer.rb +43 -60
  18. data/lib/integrity/migrations.rb +31 -48
  19. data/lib/integrity/notifier.rb +14 -16
  20. data/lib/integrity/notifier/base.rb +29 -19
  21. data/lib/integrity/notifier/test_helpers.rb +100 -0
  22. data/lib/integrity/project.rb +69 -33
  23. data/lib/integrity/project_builder.rb +23 -14
  24. data/lib/integrity/scm/git.rb +15 -14
  25. data/lib/integrity/scm/git/uri.rb +9 -9
  26. data/test/acceptance/api_test.rb +97 -0
  27. data/test/acceptance/browse_project_builds_test.rb +65 -0
  28. data/test/acceptance/browse_project_test.rb +95 -0
  29. data/test/acceptance/build_notifications_test.rb +42 -0
  30. data/test/acceptance/create_project_test.rb +97 -0
  31. data/test/acceptance/delete_project_test.rb +53 -0
  32. data/test/acceptance/edit_project_test.rb +117 -0
  33. data/test/acceptance/error_page_test.rb +18 -0
  34. data/test/acceptance/helpers.rb +2 -0
  35. data/test/acceptance/installer_test.rb +62 -0
  36. data/test/acceptance/manual_build_project_test.rb +82 -0
  37. data/test/acceptance/notifier_test.rb +109 -0
  38. data/test/acceptance/project_syndication_test.rb +30 -0
  39. data/test/acceptance/stylesheet_test.rb +18 -0
  40. data/test/helpers.rb +59 -27
  41. data/test/helpers/acceptance.rb +19 -64
  42. data/test/helpers/acceptance/email_notifier.rb +55 -0
  43. data/test/helpers/acceptance/git_helper.rb +15 -15
  44. data/test/helpers/acceptance/textfile_notifier.rb +3 -3
  45. data/test/helpers/expectations.rb +0 -1
  46. data/test/helpers/expectations/be_a.rb +4 -4
  47. data/test/helpers/expectations/change.rb +5 -5
  48. data/test/helpers/expectations/have.rb +4 -4
  49. data/test/helpers/expectations/predicates.rb +4 -4
  50. data/test/helpers/fixtures.rb +44 -18
  51. data/test/helpers/initial_migration_fixture.sql +44 -0
  52. data/test/unit/build_test.rb +51 -0
  53. data/test/unit/commit_test.rb +83 -0
  54. data/test/unit/helpers_test.rb +56 -0
  55. data/test/unit/integrity_test.rb +18 -0
  56. data/test/unit/migrations_test.rb +56 -0
  57. data/test/unit/notifier_test.rb +123 -0
  58. data/test/unit/project_builder_test.rb +108 -0
  59. data/test/unit/project_test.rb +282 -0
  60. data/test/unit/scm_test.rb +54 -0
  61. data/views/_commit_info.haml +24 -0
  62. data/views/build.haml +2 -2
  63. data/views/error.haml +4 -3
  64. data/views/home.haml +3 -5
  65. data/views/integrity.sass +19 -6
  66. data/views/new.haml +6 -6
  67. data/views/project.builder +9 -9
  68. data/views/project.haml +14 -12
  69. metadata +98 -116
  70. data/VERSION.yml +0 -4
  71. data/app.rb +0 -137
  72. data/integrity.gemspec +0 -76
  73. data/lib/integrity/core_ext/string.rb +0 -5
  74. data/test/helpers/expectations/have_tag.rb +0 -128
  75. data/views/_build_info.haml +0 -18
@@ -0,0 +1,97 @@
1
+ require File.dirname(__FILE__) + "/helpers"
2
+
3
+ class CreateProjectTest < Test::Unit::AcceptanceTestCase
4
+ story <<-EOS
5
+ As an administrator,
6
+ I want to add projects to Integrity,
7
+ So that I can know their status whenever I push code
8
+ EOS
9
+
10
+ scenario "an admin can create a public project" do
11
+ Project.first(:permalink => "integrity").should be_nil
12
+
13
+ login_as "admin", "test"
14
+
15
+ visit "/new"
16
+
17
+ fill_in "Name", :with => "Integrity"
18
+ fill_in "Git repository", :with => "git://github.com/foca/integrity.git"
19
+ fill_in "Branch to track", :with => "master"
20
+ fill_in "Build script", :with => "rake"
21
+ check "Public project"
22
+ click_button "Create Project"
23
+
24
+ Project.first(:permalink => "integrity").should_not be_nil
25
+
26
+ assert_have_tag("h1", :content => "Integrity")
27
+
28
+ log_out
29
+ visit "/integrity"
30
+
31
+ assert_have_tag("h1", :content => "Integrity")
32
+ end
33
+
34
+ scenario "an admin can create a private project" do
35
+ Project.first(:permalink => "integrity").should be_nil
36
+
37
+ login_as "admin", "test"
38
+
39
+ visit "/new"
40
+
41
+ fill_in "Name", :with => "Integrity"
42
+ fill_in "Git repository", :with => "git://github.com/foca/integrity.git"
43
+ fill_in "Branch to track", :with => "master"
44
+ fill_in "Build script", :with => "rake"
45
+ uncheck "Public project"
46
+ click_button "Create Project"
47
+
48
+ assert_have_tag("h1", :content => "Integrity")
49
+ Project.first(:permalink => "integrity").should_not be_nil
50
+
51
+ log_out
52
+ visit "/integrity"
53
+
54
+ response_code.should == 401
55
+ assert_have_tag("h1", :content => "know the password?")
56
+ end
57
+
58
+ scenario "creating a project without required fields re-renders the new project form" do
59
+ Project.first(:permalink => "integrity").should be_nil
60
+
61
+ login_as "admin", "test"
62
+
63
+ visit "/new"
64
+ click_button "Create Project"
65
+
66
+ assert_have_tag(".with_errors label", :content => "Name must not be blank")
67
+ Project.first(:permalink => "integrity").should be_nil
68
+
69
+ fill_in "Name", :with => "Integrity"
70
+ fill_in "Git repository", :with => "git://github.com/foca/integrity.git"
71
+ click_button "Create Project"
72
+
73
+ assert_have_tag("h1", :content => 'Integrity')
74
+ Project.first(:permalink => "integrity").should_not be_nil
75
+ end
76
+
77
+ scenario "a user can't see the new project form" do
78
+ visit "/new"
79
+ response_code.should == 401
80
+ assert_have_tag("h1", :content => "know the password?")
81
+ end
82
+
83
+ scenario "a user can't post the project data (bypassing the form)" do
84
+ post "/", "project_data[name]" => "Integrity",
85
+ "project_data[uri]" => "git://github.com/foca/integrity.git",
86
+ "project_data[branch]" => "master",
87
+ "project_data[command]" => "rake"
88
+
89
+ response_code.should == 401
90
+ assert_have_tag("h1", :content => "know the password?")
91
+ Project.first(:permalink => "integrity").should be_nil
92
+ end
93
+
94
+ def post(path, data={})
95
+ webrat.request_page(path, :post, data)
96
+ end
97
+ end
@@ -0,0 +1,53 @@
1
+ require File.dirname(__FILE__) + "/helpers"
2
+
3
+ class DeleteProjectTest < Test::Unit::AcceptanceTestCase
4
+ story <<-EOS
5
+ As an administrator,
6
+ I want to delete projects I don't care about anymore
7
+ So that Integrity isn't cluttered with unimportant projects
8
+ EOS
9
+
10
+ scenario "an admin can delete a project from the 'Edit Project' screen" do
11
+ Project.generate(:integrity, :commits => 4.of { Commit.gen })
12
+
13
+ login_as "admin", "test"
14
+
15
+ visit "/integrity"
16
+ click_link "Edit Project"
17
+ click_button "Yes, I'm sure, nuke it"
18
+ visit "/"
19
+
20
+ assert_have_no_tag("ul#projects", :content => "Integrity")
21
+
22
+ visit "/integrity"
23
+
24
+ response_code.should == 404
25
+ end
26
+
27
+ scenario "an admin can delete a project with an invalid SCM URI just fine" do
28
+ Project.generate(:integrity, :uri => "unknown://example.org")
29
+
30
+ login_as "admin", "test"
31
+ visit "/integrity/edit"
32
+ click_button "Yes, I'm sure, nuke it"
33
+ visit "/integrity"
34
+
35
+ response_code.should == 404
36
+ end
37
+
38
+ scenario "a user can't delete a project by doing a manual DELETE request" do
39
+ Project.gen(:integrity)
40
+
41
+ delete "/integrity"
42
+
43
+ response_code.should == 401
44
+
45
+ visit "/integrity"
46
+
47
+ assert_have_tag("h1", :content => 'Integrity')
48
+ end
49
+
50
+ def delete(path, data={})
51
+ webrat.request_page(path, :delete, data)
52
+ end
53
+ end
@@ -0,0 +1,117 @@
1
+ require File.dirname(__FILE__) + "/helpers"
2
+
3
+ class EditProjectTest < Test::Unit::AcceptanceTestCase
4
+ story <<-EOS
5
+ As an administrator,
6
+ I want to be able to edit a project
7
+ So that I can correct mistakes or update the project after a change
8
+ EOS
9
+
10
+ scenario "an admin can edit the project information" do
11
+ Project.generate(:integrity)
12
+
13
+ login_as "admin", "test"
14
+
15
+ visit "/integrity"
16
+ click_link "Edit Project"
17
+
18
+ fill_in "Name", :with => "Integrity (test refactoring)"
19
+ fill_in "Branch to track", :with => "test-refactoring"
20
+ click_button "Update Project"
21
+
22
+ assert_have_tag("h1", :content => "Integrity (test refactoring)")
23
+ end
24
+
25
+ scenario "making a public project private will hide it from the home page for non-admins" do
26
+ Project.generate(:my_test_project, :public => true)
27
+
28
+ visit "/"
29
+
30
+ assert_contain("My Test Project")
31
+
32
+ login_as "admin", "test"
33
+ visit "/my-test-project"
34
+ click_link "Edit Project"
35
+ uncheck "Public project"
36
+ click_button "Update Project"
37
+ log_out
38
+ visit "/"
39
+
40
+ assert_have_no_tag("a", :content => "My Test Project")
41
+ end
42
+
43
+ scenario "making a private project public will show it in the home page for non-admins" do
44
+ Project.generate(:my_test_project, :public => false)
45
+
46
+ visit "/"
47
+
48
+ assert_not_contain("My Test Project")
49
+
50
+ login_as "admin", "test"
51
+
52
+ visit "/my-test-project"
53
+ click_link "Edit Project"
54
+
55
+ check "Public project"
56
+ click_button "Update Project"
57
+
58
+ log_out
59
+
60
+ visit "/"
61
+
62
+ assert_have_tag("a", :content => "My Test Project")
63
+ end
64
+
65
+ scenario "a user can't edit a project's information" do
66
+ Project.generate(:integrity)
67
+
68
+ visit "/integrity"
69
+ click_link "Edit Project"
70
+
71
+ response_code.should == 401
72
+ end
73
+
74
+ scenario "an admin can see the push URL on the edit page" do
75
+ disable_auth!
76
+ Project.generate(:my_test_project)
77
+
78
+ visit "/my-test-project"
79
+ click_link "Edit Project"
80
+
81
+ assert_have_tag("#push_url", :content => "http://www.example.com/my-test-project/push")
82
+ end
83
+
84
+ scenario "public projects have a ticked 'public' checkbox on edit form" do
85
+ Project.generate(:my_test_project, :public => true)
86
+ disable_auth!
87
+ visit "/my-test-project/edit"
88
+
89
+ assert_have_tag('input[@type="checkbox"][@checked="checked"][@name="project_data[public]"]')
90
+ end
91
+
92
+ scenario "private projects have an unticked 'public' checkbox on edit form" do
93
+ Project.generate(:my_test_project, :public => false)
94
+ disable_auth!
95
+ visit "/my-test-project/edit"
96
+
97
+ assert_have_no_tag('input[@type="checkbox"][@checked][@name="project_data[public]"]')
98
+ end
99
+
100
+ scenario "after I uncheck the public checkbox, it should still be uncheck after I save" do
101
+ Project.generate(:integrity, :public => true)
102
+
103
+ login_as "admin", "test"
104
+
105
+ visit "/integrity"
106
+ click_link "Edit Project"
107
+
108
+ assert_have_tag('input[@type="checkbox"][@checked="checked"][@name="project_data[public]"]')
109
+
110
+ uncheck "project_public"
111
+ click_button "Update Project"
112
+
113
+ click_link "Edit Project"
114
+
115
+ assert_have_no_tag('input[@type="checkbox"][@checked="checked"][@name="project_data[public]"]')
116
+ end
117
+ end
@@ -0,0 +1,18 @@
1
+ require File.dirname(__FILE__) + "/helpers"
2
+
3
+ class ErrorPageTest < Test::Unit::AcceptanceTestCase
4
+ story <<-EOS
5
+ As an user,
6
+ I want to be shown a friendly page when something go terribly wrong
7
+ So that I can understand what's going on
8
+ EOS
9
+
10
+ scenario "an error happen while I am browsing my Integrity install" do
11
+ stub(Project).only_public_unless(false) { raise ArgumentError }
12
+ lambda { visit "/" }.should raise_error(Webrat::PageLoadError)
13
+
14
+ response_code.should == 500
15
+ assert_have_tag("h1", :content => "Whatever you do")
16
+ assert_have_tag("strong", :content => "ArgumentError")
17
+ end
18
+ end
@@ -0,0 +1,2 @@
1
+ require File.dirname(__FILE__) + "/../helpers"
2
+ require File.dirname(__FILE__) / ".." / "helpers" / "acceptance"
@@ -0,0 +1,62 @@
1
+ require File.dirname(__FILE__) + "/helpers"
2
+ require "integrity/installer"
3
+
4
+ class InstallerTest < Test::Unit::AcceptanceTestCase
5
+ include FileUtils
6
+
7
+ story <<-EOS
8
+ As an user,
9
+ I want to easily install Integrity
10
+ So that I can spend time actually writing code
11
+ EOS
12
+
13
+ before(:each) do
14
+ rm_rf root if File.directory?(root)
15
+ end
16
+
17
+ def root
18
+ Pathname("/tmp/i-haz-integrity")
19
+ end
20
+
21
+ def install(options={})
22
+ installer = Installer.new
23
+ installer.options = { :passenger => false, :thin => false }.merge!(options)
24
+ stdout, _ = util_capture { installer.install(root.to_s) }
25
+ stdout
26
+ end
27
+
28
+ scenario "Installing integrity into a given directory" do
29
+ assert install.include?("Awesome")
30
+
31
+ assert root.join("builds").directory?
32
+ assert root.join("log").directory?
33
+ assert ! root.join("public").directory?
34
+ assert ! root.join("tmp").directory?
35
+
36
+ assert ! root.join("thin.yml").file?
37
+ assert root.join("config.ru").file?
38
+
39
+ config = YAML.load_file(root.join("config.yml"))
40
+
41
+ config[:export_directory].should == root.join("builds").to_s
42
+ config[:database_uri].should == "sqlite3://#{root}/integrity.db"
43
+ config[:log].should == root.join("log/integrity.log").to_s
44
+ end
45
+
46
+ scenario "Installing integrity for Passenger" do
47
+ install(:passenger => true)
48
+
49
+ assert root.join("public").directory?
50
+ assert root.join("tmp").directory?
51
+ end
52
+
53
+ scenario "Installing Integrity for Thin" do
54
+ install(:thin => true)
55
+
56
+ config = YAML.load_file(root.join("thin.yml"))
57
+ config["chdir"].should == root.to_s
58
+ config["pid"].should == root.join("thin.pid").to_s
59
+ config["rackup"].should == root.join("config.ru").to_s
60
+ config["log"].should == root.join("log/thin.log").to_s
61
+ end
62
+ end
@@ -0,0 +1,82 @@
1
+ require File.dirname(__FILE__) + "/helpers"
2
+
3
+ class ManualBuildProjectTest < Test::Unit::AcceptanceTestCase
4
+ story <<-EOS
5
+ As an administrator,
6
+ I want to manually build my project
7
+ So that I know if it builds properly
8
+ EOS
9
+
10
+ scenario "clicking on 'Manual Build' triggers a successful build" do
11
+ git_repo(:my_test_project).add_successful_commit
12
+ Project.gen(:my_test_project, :uri => git_repo(:my_test_project).path)
13
+ login_as "admin", "test"
14
+
15
+ visit "/my-test-project"
16
+ click_button "manual build"
17
+
18
+ assert_have_tag("h1", :content =>
19
+ "Built #{git_repo(:my_test_project).short_head} successfully")
20
+ assert_have_tag("blockquote p", :content => "This commit will work")
21
+ assert_have_tag("span.who", :content => "by: John Doe")
22
+ assert_have_tag("span.when", :content => "today")
23
+ assert_have_tag("pre.output", :content => "Running tests...")
24
+ end
25
+
26
+ scenario "clicking on 'Manual Build' triggers a failed build" do
27
+ git_repo(:my_test_project).add_failing_commit
28
+ Project.gen(:my_test_project, :uri => git_repo(:my_test_project).path)
29
+ login_as "admin", "test"
30
+
31
+ visit "/my-test-project"
32
+ click_button "manual build"
33
+
34
+ assert_have_tag("h1",
35
+ :content => "Built #{git_repo(:my_test_project).short_head} and failed")
36
+ assert_have_tag("blockquote p", :content => "This commit will fail")
37
+ end
38
+
39
+ scenario "fixing the build command and then rebuilding result in a successful build" do
40
+ git_repo(:my_test_project).add_successful_commit
41
+ Project.gen(:my_test_project,
42
+ :uri => git_repo(:my_test_project).path,
43
+ :command => "ruby not-found.rb")
44
+
45
+ login_as "admin", "test"
46
+
47
+ visit "/my-test-project"
48
+ click_button "manual build"
49
+ assert_have_tag("h1", :content => "failed")
50
+
51
+ visit "/my-test-project/edit"
52
+ fill_in "Build script", :with => "./test"
53
+ click_button "Update Project"
54
+
55
+ visit "/my-test-project"
56
+ click_button "Build the last commit"
57
+
58
+ assert_have_tag("h1", :content => "success")
59
+ end
60
+
61
+ scenario "Successful builds should not display the 'Rebuild' button" do
62
+ git_repo(:my_test_project).add_successful_commit
63
+ Project.gen(:my_test_project, :uri => git_repo(:my_test_project).path)
64
+ login_as "admin", "test"
65
+
66
+ visit "/my-test-project"
67
+ click_button "manual build"
68
+
69
+ assert_have_no_tag("button", :content => "Rebuild")
70
+ end
71
+
72
+ scenario "Failed builds should display the 'Rebuild' button" do
73
+ git_repo(:my_test_project).add_failing_commit
74
+ Project.gen(:my_test_project, :uri => git_repo(:my_test_project).path)
75
+ login_as "admin", "test"
76
+
77
+ visit "/my-test-project"
78
+ click_button "manual build"
79
+
80
+ assert_have_tag("button", :content => "Rebuild")
81
+ end
82
+ end
@@ -0,0 +1,109 @@
1
+ require File.dirname(__FILE__) + "/helpers"
2
+
3
+ class NotifierConfigIssues < Test::Unit::AcceptanceTestCase
4
+ story <<-EOS
5
+ As an administrator,
6
+ I want to add multiple projects to Integrity,
7
+ So that I can be certain notifiers remain functional (cf #43)
8
+ EOS
9
+
10
+ before(:each) do
11
+ # This is needed before any available notifier is unset
12
+ # in the global #before
13
+ load File.dirname(__FILE__) + "/../helpers/acceptance/email_notifier.rb"
14
+ end
15
+
16
+ def fill_in_email_notifier
17
+ fill_in "notifiers[Email][to]", :with => "quentin@example.com"
18
+ fill_in "notifiers[Email][from]", :with => "ci@example.com"
19
+ fill_in "notifiers[Email][user]", :with => "inspector"
20
+ fill_in "notifiers[Email][pass]", :with => "gadget"
21
+ fill_in "notifiers[Email][auth]", :with => "simple"
22
+ fill_in "notifiers[Email][domain]", :with => "example.com"
23
+ end
24
+
25
+ def fill_in_project_info(name, repo)
26
+ fill_in "Name", :with => name
27
+ fill_in "Git repository", :with => repo
28
+ fill_in "Branch to track", :with => "master"
29
+ fill_in "Build script", :with => "rake"
30
+ check "Public project"
31
+
32
+ fill_in_email_notifier
33
+ end
34
+
35
+ def assert_have_email_notifier
36
+ assert_have_tag "input#email_notifier_to[@value='quentin@example.com']"
37
+ assert_have_tag "input#email_notifier_from[@value='ci@example.com']"
38
+ assert_have_tag "input#email_notifier_user[@value='inspector']"
39
+ assert_have_tag "input#email_notifier_pass[@value='gadget']"
40
+ assert_have_tag "input#email_notifier_auth[@value='simple']"
41
+ assert_have_tag "input#email_notifier_domain[@value='example.com']"
42
+ end
43
+
44
+ def add_project(name, repo)
45
+ visit "/new"
46
+ fill_in_project_info(name, repo)
47
+ click_button "Create Project"
48
+
49
+ assert_have_tag("h1", :content => name)
50
+ click_link 'Edit Project'
51
+ assert_have_email_notifier
52
+ end
53
+
54
+ def edit_project(name)
55
+ visit "/#{name}"
56
+ click_link "Edit Project"
57
+ assert_have_email_notifier
58
+ fill_in :branch, :with => "testing"
59
+ click_button "Update Project"
60
+ end
61
+
62
+ scenario "an admin can create a public project and retain mailer info" do
63
+ Project.first(:permalink => "integrity").should be_nil
64
+
65
+ login_as "admin", "test"
66
+
67
+ visit "/"
68
+ add_project "Integrity", "git://github.com/foca/integrity.git"
69
+ edit_project "integrity"
70
+
71
+ visit "/integrity"
72
+ click_link "Edit Project"
73
+
74
+ assert_have_email_notifier
75
+ end
76
+
77
+ scenario "an admin can create multiple public projects" do
78
+ Project.first(:permalink => "integrity").should be_nil
79
+
80
+ login_as "admin", "test"
81
+
82
+ visit "/"
83
+
84
+ add_project "Integrity", "git://github.com/foca/integrity.git"
85
+ click_link "projects"
86
+
87
+ add_project "Webrat", "git://github.com/brynary/webrat.git"
88
+ click_link "projects"
89
+
90
+ add_project "Rails", "git://github.com/rails/rails.git"
91
+ click_link "projects"
92
+
93
+ edit_project "integrity"
94
+ edit_project "webrat"
95
+ edit_project "rails"
96
+
97
+ visit "/integrity"
98
+ click_link "Edit Project"
99
+ assert_have_email_notifier
100
+
101
+ visit "/webrat"
102
+ click_link "Edit Project"
103
+ assert_have_email_notifier
104
+
105
+ visit "/rails"
106
+ click_link "Edit Project"
107
+ assert_have_email_notifier
108
+ end
109
+ end