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 (77) hide show
  1. data/README.markdown +7 -0
  2. data/Rakefile +77 -124
  3. data/config/config.ru +29 -0
  4. data/config/config.sample.ru +6 -16
  5. data/config/config.sample.yml +15 -12
  6. data/config/config.yml +34 -0
  7. data/lib/integrity.rb +13 -13
  8. data/lib/integrity/app.rb +138 -0
  9. data/lib/integrity/author.rb +39 -0
  10. data/lib/integrity/build.rb +54 -31
  11. data/lib/integrity/commit.rb +71 -0
  12. data/lib/integrity/helpers.rb +3 -3
  13. data/lib/integrity/helpers/authorization.rb +2 -2
  14. data/lib/integrity/helpers/forms.rb +3 -3
  15. data/lib/integrity/helpers/pretty_output.rb +1 -1
  16. data/lib/integrity/helpers/rendering.rb +6 -1
  17. data/lib/integrity/helpers/resources.rb +9 -3
  18. data/lib/integrity/helpers/urls.rb +15 -13
  19. data/lib/integrity/installer.rb +43 -60
  20. data/lib/integrity/migrations.rb +31 -48
  21. data/lib/integrity/notifier.rb +14 -16
  22. data/lib/integrity/notifier/base.rb +29 -19
  23. data/lib/integrity/notifier/test_helpers.rb +100 -0
  24. data/lib/integrity/project.rb +69 -33
  25. data/lib/integrity/project_builder.rb +23 -14
  26. data/lib/integrity/scm/git.rb +15 -14
  27. data/lib/integrity/scm/git/uri.rb +9 -9
  28. data/test/acceptance/api_test.rb +97 -0
  29. data/test/acceptance/browse_project_builds_test.rb +65 -0
  30. data/test/acceptance/browse_project_test.rb +95 -0
  31. data/test/acceptance/build_notifications_test.rb +42 -0
  32. data/test/acceptance/create_project_test.rb +97 -0
  33. data/test/acceptance/delete_project_test.rb +53 -0
  34. data/test/acceptance/edit_project_test.rb +117 -0
  35. data/test/acceptance/error_page_test.rb +18 -0
  36. data/test/acceptance/helpers.rb +2 -0
  37. data/test/acceptance/installer_test.rb +62 -0
  38. data/test/acceptance/manual_build_project_test.rb +82 -0
  39. data/test/acceptance/notifier_test.rb +109 -0
  40. data/test/acceptance/project_syndication_test.rb +30 -0
  41. data/test/acceptance/stylesheet_test.rb +18 -0
  42. data/test/helpers.rb +59 -26
  43. data/test/helpers/acceptance.rb +19 -65
  44. data/test/helpers/acceptance/email_notifier.rb +55 -0
  45. data/test/helpers/acceptance/git_helper.rb +15 -15
  46. data/test/helpers/acceptance/textfile_notifier.rb +3 -3
  47. data/test/helpers/expectations.rb +0 -1
  48. data/test/helpers/expectations/be_a.rb +4 -4
  49. data/test/helpers/expectations/change.rb +5 -5
  50. data/test/helpers/expectations/have.rb +4 -4
  51. data/test/helpers/expectations/predicates.rb +4 -4
  52. data/test/helpers/fixtures.rb +44 -18
  53. data/test/helpers/initial_migration_fixture.sql +44 -0
  54. data/test/unit/build_test.rb +51 -0
  55. data/test/unit/commit_test.rb +83 -0
  56. data/test/unit/helpers_test.rb +56 -0
  57. data/test/unit/integrity_test.rb +18 -0
  58. data/test/unit/migrations_test.rb +56 -0
  59. data/test/unit/notifier_test.rb +123 -0
  60. data/test/unit/project_builder_test.rb +108 -0
  61. data/test/unit/project_test.rb +282 -0
  62. data/test/unit/scm_test.rb +54 -0
  63. data/views/_commit_info.haml +24 -0
  64. data/views/build.haml +2 -2
  65. data/views/error.haml +4 -3
  66. data/views/home.haml +3 -5
  67. data/views/integrity.sass +19 -6
  68. data/views/new.haml +6 -6
  69. data/views/project.builder +9 -9
  70. data/views/project.haml +14 -12
  71. metadata +89 -122
  72. data/VERSION.yml +0 -4
  73. data/app.rb +0 -138
  74. data/integrity.gemspec +0 -76
  75. data/lib/integrity/core_ext/string.rb +0 -5
  76. data/test/helpers/expectations/have_tag.rb +0 -128
  77. data/views/_build_info.haml +0 -18
@@ -0,0 +1,95 @@
1
+ require File.dirname(__FILE__) + "/helpers"
2
+
3
+ class BrowsePublicProjectsTest < Test::Unit::AcceptanceTestCase
4
+ story <<-EOS
5
+ As a user,
6
+ I want to browse public projects on Integrity,
7
+ So I can follow the status of my favorite OSS projects
8
+ EOS
9
+
10
+ scenario "a user can see a public project listed on the home page" do
11
+ Project.gen(:integrity, :public => true)
12
+ Project.gen(:my_test_project, :public => true)
13
+
14
+ visit "/"
15
+
16
+ assert_have_tag("a", :content => "Integrity")
17
+ assert_have_tag("a", :content => "My Test Project")
18
+ end
19
+
20
+ scenario "a user can't see a private project listed on the home page" do
21
+ Project.gen(:my_test_project, :public => false)
22
+ Project.gen(:integrity, :public => true)
23
+
24
+ visit "/"
25
+
26
+ assert_have_no_tag("a", :content => "My Test Project")
27
+ assert_have_tag("a", :content => "Integrity")
28
+ end
29
+
30
+ scenario "a user can see the projects status on the home page" do
31
+ integrity = Project.gen(:integrity, :commits => 3.of { Commit.gen(:successful) })
32
+ test = Project.gen(:my_test_project, :commits => 2.of { Commit.gen(:failed) })
33
+ no_build = Project.gen(:public => true, :building => false)
34
+ building = Project.gen(:public => true, :building => true)
35
+
36
+ visit "/"
37
+
38
+ assert_contain("Built #{integrity.last_commit.short_identifier} successfully")
39
+ assert_contain("Built #{test.last_commit.short_identifier} and failed")
40
+ assert_contain("Never built yet")
41
+ assert_contain("Building!")
42
+ end
43
+
44
+ scenario "a user clicking through a link on the home page for a public project arrives at the project page" do
45
+ Project.gen(:my_test_project, :public => true)
46
+
47
+ visit "/"
48
+ click_link "My Test Project"
49
+
50
+ assert_have_tag("h1", :content => "My Test Project")
51
+ end
52
+
53
+ scenario "a user gets a 404 when browsing to an unexisting project" do
54
+ visit "/who-are-you"
55
+
56
+ response_code.should == 404
57
+ assert_have_tag("h1", :content => "you seem a bit lost, sir")
58
+ end
59
+
60
+ scenario "a user browsing to the url of a private project gets a 401" do
61
+ Project.gen(:my_test_project, :public => false)
62
+
63
+ visit "/my-test-project"
64
+
65
+ response_code.should == 401
66
+ assert_have_tag("h1", :content => "know the password?")
67
+ end
68
+
69
+ scenario "an admin can browse to a private project just fine" do
70
+ Project.gen(:my_test_project, :public => false)
71
+
72
+ login_as "admin", "test"
73
+
74
+ visit "/"
75
+ click_link "My Test Project"
76
+
77
+ assert_have_tag("h1", :content => "My Test Project")
78
+ end
79
+
80
+ scenario "a user browsing to a public project with no build see a friendly message" do
81
+ project = Project.gen(:my_test_project, :public => true)
82
+
83
+ visit "/my-test-project"
84
+ assert_contain("No builds for this project, buddy")
85
+ end
86
+
87
+ scenario "an admin browsing to a private project with no build see a friendly message" do
88
+ Project.gen(:my_test_project, :public => false)
89
+
90
+ login_as "admin", "test"
91
+ visit "/my-test-project"
92
+
93
+ assert_contain("No builds for this project, buddy")
94
+ end
95
+ end
@@ -0,0 +1,42 @@
1
+ require File.dirname(__FILE__) + "/helpers"
2
+ require File.dirname(__FILE__) + "/../helpers/acceptance/textfile_notifier"
3
+
4
+ class BuildNotificationsTest < Test::Unit::AcceptanceTestCase
5
+ story <<-EOS
6
+ As an administrator,
7
+ I want to setup notifiers on my projects
8
+ So that I get alerts with every build
9
+ EOS
10
+
11
+ before(:each) do
12
+ # This is needed before any available notifier is unset
13
+ # in the global #before
14
+ load File.dirname(__FILE__) + "/../helpers/acceptance/textfile_notifier.rb"
15
+ end
16
+
17
+ scenario "an admin sets up a notifier for a project that didn't have any" do
18
+ git_repo(:my_test_project).add_successful_commit
19
+ Project.gen(:my_test_project, :notifiers => [], :uri => git_repo(:my_test_project).path)
20
+ rm_f "/tmp/textfile_notifications.txt"
21
+
22
+ login_as "admin", "test"
23
+
24
+ visit "/my-test-project"
25
+
26
+ click_link "Edit Project"
27
+ check "enabled_notifiers_textfile"
28
+ fill_in "File", :with => "/tmp/textfile_notifications.txt"
29
+ click_button "Update Project"
30
+
31
+ click_button "manual build"
32
+
33
+ notification = File.read("/tmp/textfile_notifications.txt")
34
+ notification.should =~ /=== Built #{git_repo(:my_test_project).short_head} successfully ===/
35
+ notification.should =~ /Build #{git_repo(:my_test_project).head} was successful/
36
+ notification.should =~ %r(http://www.example.com/my-test-project/commits/#{git_repo(:my_test_project).head})
37
+ notification.should =~ /Commit Author: John Doe/
38
+ notification.should =~ /Commit Date: (.+)/
39
+ notification.should =~ /Commit Message: This commit will work/
40
+ notification.should =~ /Build Output:\n\nRunning tests...\n/
41
+ end
42
+ end
@@ -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