houston-core 0.5.0.beta1 → 0.5.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.codeclimate.yml +0 -2
- data/.gitignore +3 -0
- data/Gemfile +1 -0
- data/Gemfile.lock +63 -62
- data/README.md +13 -121
- data/app/adapters/houston/adapters/ci_server/errors.rb +4 -4
- data/app/adapters/houston/adapters/ci_server/jenkins_adapter/job.rb +50 -50
- data/app/adapters/houston/adapters/ci_server/jenkins_adapter.rb +4 -4
- data/app/adapters/houston/adapters/ci_server/none_adapter/job.rb +5 -5
- data/app/adapters/houston/adapters/ci_server/none_adapter.rb +4 -4
- data/app/adapters/houston/adapters/deployment/engineyard.rb +20 -20
- data/app/adapters/houston/adapters/error_tracker/errbit_adapter/app.rb +14 -14
- data/app/adapters/houston/adapters/error_tracker/errbit_adapter/connection.rb +41 -41
- data/app/adapters/houston/adapters/error_tracker/errbit_adapter/notice.rb +6 -6
- data/app/adapters/houston/adapters/error_tracker/errbit_adapter/problem.rb +25 -25
- data/app/adapters/houston/adapters/error_tracker/errbit_adapter.rb +20 -20
- data/app/adapters/houston/adapters/error_tracker/none_adapter.rb +8 -8
- data/app/adapters/houston/adapters/error_tracker/null_app.rb +13 -13
- data/app/adapters/houston/adapters/ticket_tracker/errors.rb +5 -5
- data/app/adapters/houston/adapters/ticket_tracker/github_adapter/connection.rb +41 -41
- data/app/adapters/houston/adapters/ticket_tracker/github_adapter/issue.rb +22 -22
- data/app/adapters/houston/adapters/ticket_tracker/github_adapter.rb +7 -7
- data/app/adapters/houston/adapters/ticket_tracker/houston_adapter/connection.rb +13 -13
- data/app/adapters/houston/adapters/ticket_tracker/houston_adapter/ticket.rb +15 -15
- data/app/adapters/houston/adapters/ticket_tracker/houston_adapter.rb +4 -4
- data/app/adapters/houston/adapters/ticket_tracker/none_adapter.rb +4 -4
- data/app/adapters/houston/adapters/ticket_tracker/null_connection.rb +13 -13
- data/app/adapters/houston/adapters/ticket_tracker/null_ticket.rb +19 -19
- data/app/adapters/houston/adapters/ticket_tracker/unfuddle_adapter/connection.rb +54 -54
- data/app/adapters/houston/adapters/ticket_tracker/unfuddle_adapter/milestone.rb +10 -10
- data/app/adapters/houston/adapters/ticket_tracker/unfuddle_adapter/ticket.rb +59 -59
- data/app/adapters/houston/adapters/ticket_tracker/unfuddle_adapter.rb +9 -9
- data/app/adapters/houston/adapters/version_control/commit.rb +4 -4
- data/app/adapters/houston/adapters/version_control/errors.rb +9 -9
- data/app/adapters/houston/adapters/version_control/git_adapter/github_repo.rb +15 -15
- data/app/adapters/houston/adapters/version_control/git_adapter/remote_repo.rb +53 -30
- data/app/adapters/houston/adapters/version_control/git_adapter/repo.rb +60 -55
- data/app/adapters/houston/adapters/version_control/git_adapter.rb +22 -22
- data/app/adapters/houston/adapters/version_control/none_adapter.rb +4 -4
- data/app/adapters/houston/adapters/version_control/null_commit.rb +4 -4
- data/app/adapters/houston/adapters/version_control/null_repo.rb +18 -18
- data/app/adapters/houston/adapters.rb +10 -10
- data/app/assets/javascripts/app/boot.coffee +5 -5
- data/app/assets/javascripts/app/infinite_scroll.coffee +6 -6
- data/app/assets/javascripts/app/models/commit.coffee +1 -1
- data/app/assets/javascripts/app/models/release.coffee +1 -1
- data/app/assets/javascripts/app/models/task.coffee +1 -1
- data/app/assets/javascripts/app/models/tester.coffee +1 -1
- data/app/assets/javascripts/app/models/testing_note.coffee +2 -2
- data/app/assets/javascripts/app/models/ticket.coffee +24 -24
- data/app/assets/javascripts/app/models/user.coffee +1 -1
- data/app/assets/javascripts/app/releases.coffee +6 -6
- data/app/assets/javascripts/app/stacked_area_graph.coffee +22 -22
- data/app/assets/javascripts/app/stacked_bar_graph.coffee +20 -20
- data/app/assets/javascripts/app/table_row_expander.coffee +10 -10
- data/app/assets/javascripts/app/ticket_tracker_refresh.coffee +19 -19
- data/app/assets/javascripts/app/views/_show_sprint_view.coffee +14 -14
- data/app/assets/javascripts/app/views/_tickets_view.coffee +8 -8
- data/app/assets/javascripts/app/views/all_tickets_view.coffee +1 -2
- data/app/assets/javascripts/app/views/commit_view.coffee +2 -2
- data/app/assets/javascripts/app/views/edit_sprint_view.coffee +36 -36
- data/app/assets/javascripts/app/views/find_or_create_ticket_view.coffee +3 -3
- data/app/assets/javascripts/app/views/keyboard_shortcuts_modal.coffee +14 -14
- data/app/assets/javascripts/app/views/new_ticket_modal.coffee +4 -4
- data/app/assets/javascripts/app/views/new_ticket_view.coffee +34 -34
- data/app/assets/javascripts/app/views/problems_view.coffee +8 -8
- data/app/assets/javascripts/app/views/reports_view.coffee +5 -5
- data/app/assets/javascripts/app/views/testing_note_view.coffee +10 -10
- data/app/assets/javascripts/app/views/testing_report_view.coffee +7 -7
- data/app/assets/javascripts/app/views/testing_ticket_view.coffee +36 -36
- data/app/assets/javascripts/app/views/ticket_modal_view.coffee +5 -5
- data/app/assets/javascripts/core/app.coffee +22 -22
- data/app/assets/javascripts/core/burndown_chart.coffee +21 -21
- data/app/assets/javascripts/core/errors.coffee +2 -2
- data/app/assets/javascripts/core/handlebars_helpers.coffee +6 -6
- data/app/assets/javascripts/core/jquery_extensions.coffee +28 -28
- data/app/assets/javascripts/dashboard/refresher.coffee +4 -4
- data/app/assets/stylesheets/application/ansi.scss +2 -2
- data/app/assets/stylesheets/application/commit.scss +1 -1
- data/app/assets/stylesheets/application/exceptions.scss +3 -3
- data/app/assets/stylesheets/application/find_or_create_ticket.scss +2 -2
- data/app/assets/stylesheets/application/follow_up.scss +2 -2
- data/app/assets/stylesheets/application/forms.scss +2 -2
- data/app/assets/stylesheets/application/freight_train.css.scss +5 -5
- data/app/assets/stylesheets/application/full_screen.scss +2 -2
- data/app/assets/stylesheets/application/infinite_scroll.scss +1 -1
- data/app/assets/stylesheets/application/keyboard_shortcuts.scss +3 -3
- data/app/assets/stylesheets/application/markdown.scss +2 -2
- data/app/assets/stylesheets/application/navigation.scss +15 -15
- data/app/assets/stylesheets/application/new_ticket_view.scss +16 -16
- data/app/assets/stylesheets/application/omnibar.scss +6 -6
- data/app/assets/stylesheets/application/project_banner_buttons.scss +6 -6
- data/app/assets/stylesheets/application/project_tiles.scss +2 -2
- data/app/assets/stylesheets/application/projects.css.scss +3 -3
- data/app/assets/stylesheets/application/pull_requests.scss +6 -6
- data/app/assets/stylesheets/application/queue.scss +2 -2
- data/app/assets/stylesheets/application/release_form.scss +10 -10
- data/app/assets/stylesheets/application/releases.scss +32 -39
- data/app/assets/stylesheets/application/sortable_table.scss +6 -6
- data/app/assets/stylesheets/application/sprint.scss +6 -6
- data/app/assets/stylesheets/application/tables.scss +6 -6
- data/app/assets/stylesheets/application/test_run.scss +16 -16
- data/app/assets/stylesheets/application/ticket.scss +13 -13
- data/app/assets/stylesheets/application/ticket_modal.scss +19 -19
- data/app/assets/stylesheets/application/tickets.scss +2 -2
- data/app/assets/stylesheets/application/timeline.scss +30 -30
- data/app/assets/stylesheets/application/tips.scss +1 -1
- data/app/assets/stylesheets/application/typeahead.scss +4 -4
- data/app/assets/stylesheets/application/uploading.scss +13 -13
- data/app/assets/stylesheets/application/welcome.scss +3 -3
- data/app/assets/stylesheets/core/burndown_chart.scss +11 -11
- data/app/assets/stylesheets/core/misc.scss +8 -8
- data/app/assets/stylesheets/core/overrides.scss +2 -2
- data/app/assets/stylesheets/core/scores.scss +26 -26
- data/app/assets/stylesheets/dashboard/dashboard.scss +16 -16
- data/app/assets/stylesheets/print.css.scss +3 -3
- data/app/concerns/commit_synchronizer.rb +24 -24
- data/app/concerns/feature_state.rb +7 -7
- data/app/concerns/feature_support.rb +4 -4
- data/app/concerns/historical_weekly_stats.rb +4 -4
- data/app/concerns/milestone_synchronizer.rb +14 -14
- data/app/concerns/nosync.rb +4 -4
- data/app/concerns/project_adapter.rb +22 -22
- data/app/concerns/retirement.rb +7 -7
- data/app/concerns/ticket_synchronizer.rb +31 -31
- data/app/concerns/unique_add.rb +2 -2
- data/app/controllers/api/v1/sprint_tasks_controller.rb +19 -19
- data/app/controllers/api/v1/ticket_tasks_controller.rb +14 -14
- data/app/controllers/application_controller.rb +46 -40
- data/app/controllers/commits_controller.rb +10 -10
- data/app/controllers/deploys_controller.rb +11 -11
- data/app/controllers/home_controller.rb +4 -4
- data/app/controllers/jobs_controller.rb +3 -3
- data/app/controllers/oauth_consumers_controller.rb +1 -1
- data/app/controllers/omnibar_controller.rb +6 -6
- data/app/controllers/project_exceptions_controller.rb +11 -11
- data/app/controllers/project_hooks_controller.rb +7 -7
- data/app/controllers/project_options_controller.rb +8 -8
- data/app/controllers/project_pretickets_controller.rb +6 -6
- data/app/controllers/project_roles_controller.rb +7 -7
- data/app/controllers/project_tickets_controller.rb +28 -28
- data/app/controllers/project_tickets_sync_controller.rb +8 -8
- data/app/controllers/projects_controller.rb +29 -29
- data/app/controllers/releases_controller.rb +25 -25
- data/app/controllers/reports_controller.rb +36 -36
- data/app/controllers/sessions_controller.rb +2 -2
- data/app/controllers/settings_controller.rb +3 -3
- data/app/controllers/sprint_task_locks_controller.rb +11 -11
- data/app/controllers/sprints_controller.rb +25 -25
- data/app/controllers/tasks_controller.rb +10 -10
- data/app/controllers/test_runs_controller.rb +8 -8
- data/app/controllers/tester_bar_controller.rb +2 -2
- data/app/controllers/testing_notes_controller.rb +15 -15
- data/app/controllers/testing_report_controller.rb +14 -14
- data/app/controllers/tickets_controller.rb +9 -9
- data/app/controllers/uploads_controller.rb +9 -9
- data/app/controllers/user_credentials_controller.rb +4 -4
- data/app/controllers/user_options_controller.rb +6 -6
- data/app/controllers/users_controller.rb +34 -34
- data/app/controllers/welcome_controller.rb +6 -6
- data/app/helpers/ansi_helper.rb +4 -4
- data/app/helpers/application_helper.rb +26 -26
- data/app/helpers/avatar_helper.rb +12 -12
- data/app/helpers/backtrace_helper.rb +2 -2
- data/app/helpers/breadcrumbs_helper.rb +9 -9
- data/app/helpers/commit_helper.rb +16 -12
- data/app/helpers/demo_helper.rb +4 -4
- data/app/helpers/email_helper.rb +3 -3
- data/app/helpers/emoji_helper.rb +1 -1
- data/app/helpers/excel_helpers.rb +2 -2
- data/app/helpers/exposure_helper.rb +3 -3
- data/app/helpers/flash_message_helper.rb +6 -6
- data/app/helpers/maintenance_light_helper.rb +3 -3
- data/app/helpers/markdown_helper.rb +3 -3
- data/app/helpers/navigation_helper.rb +12 -12
- data/app/helpers/project_helper.rb +5 -5
- data/app/helpers/release_helper.rb +9 -9
- data/app/helpers/score_card_helper.rb +16 -16
- data/app/helpers/sprint_helper.rb +2 -2
- data/app/helpers/static_chart_helper.rb +2 -2
- data/app/helpers/test_run_helper.rb +6 -6
- data/app/helpers/ticket_helper.rb +14 -14
- data/app/helpers/timeline_helper.rb +5 -5
- data/app/helpers/url_helper.rb +26 -26
- data/app/helpers/user_helper.rb +6 -6
- data/app/helpers/version_helper.rb +4 -4
- data/app/interactors/test_run_comparer.rb +9 -9
- data/app/interactors/ticket_report.rb +7 -7
- data/app/jobs/sync_all_tickets_job.rb +13 -13
- data/app/jobs/sync_commits_job.rb +3 -3
- data/app/jobs/sync_project_tickets_job.rb +9 -9
- data/app/mailers/deploy_notification.rb +9 -9
- data/app/mailers/project_notification.rb +29 -29
- data/app/mailers/view_mailer.rb +20 -38
- data/app/models/ability.rb +21 -21
- data/app/models/activity_feed.rb +25 -25
- data/app/models/antecedent.rb +3 -3
- data/app/models/code_climate/coverage_report.rb +25 -25
- data/app/models/commit.rb +77 -82
- data/app/models/consumer_token.rb +2 -2
- data/app/models/deploy.rb +31 -27
- data/app/models/environment.rb +12 -12
- data/app/models/github/commit_status_report.rb +16 -16
- data/app/models/github/pull_request.rb +35 -12
- data/app/models/github/pull_request_event.rb +6 -5
- data/app/models/github_token.rb +2 -2
- data/app/models/historical_head.rb +2 -2
- data/app/models/key_dependency.rb +17 -17
- data/app/models/maintenance_light.rb +3 -3
- data/app/models/measurement.rb +22 -22
- data/app/models/menu_item.rb +5 -5
- data/app/models/menu_item_divider.rb +1 -1
- data/app/models/milestone.rb +27 -27
- data/app/models/output_stream.rb +4 -4
- data/app/models/post_receive_payload.rb +7 -7
- data/app/models/project.rb +93 -89
- data/app/models/project_dependencies.rb +9 -9
- data/app/models/project_dependency.rb +33 -33
- data/app/models/project_menu_item.rb +3 -3
- data/app/models/release.rb +60 -60
- data/app/models/release_change.rb +18 -18
- data/app/models/role.rb +15 -15
- data/app/models/rubygems/gem.rb +21 -21
- data/app/models/run_tests_on_post_receive.rb +36 -36
- data/app/models/setting.rb +3 -3
- data/app/models/settings.rb +8 -8
- data/app/models/source_file_coverage.rb +2 -2
- data/app/models/sprint.rb +19 -19
- data/app/models/sprint_report.rb +7 -7
- data/app/models/sprint_task.rb +8 -8
- data/app/models/static_chart/area.rb +19 -19
- data/app/models/static_chart.rb +35 -35
- data/app/models/task.rb +64 -64
- data/app/models/test_error.rb +2 -2
- data/app/models/test_result.rb +7 -0
- data/app/models/test_run.rb +75 -79
- data/app/models/test_run_statistics.rb +18 -19
- data/app/models/testing_note.rb +19 -19
- data/app/models/ticket.rb +104 -104
- data/app/models/ticket_antecedent.rb +11 -11
- data/app/models/ticket_comment.rb +6 -6
- data/app/models/ticket_committer.rb +3 -3
- data/app/models/ticket_tag.rb +6 -6
- data/app/models/user.rb +44 -44
- data/app/models/user_credentials.rb +11 -11
- data/app/models/value_statement.rb +3 -3
- data/app/presenters/commit_presenter.rb +6 -6
- data/app/presenters/one_or_many.rb +4 -4
- data/app/presenters/problem_presenter.rb +8 -8
- data/app/presenters/release_presenter.rb +4 -4
- data/app/presenters/sprint_task_presenter.rb +9 -9
- data/app/presenters/task_presenter.rb +8 -8
- data/app/presenters/tasks_excel_presenter.rb +6 -6
- data/app/presenters/tester_presenter.rb +3 -3
- data/app/presenters/testing_note_presenter.rb +4 -4
- data/app/presenters/testing_report_ticket_presenter.rb +9 -9
- data/app/presenters/ticket_presenter.rb +4 -4
- data/app/views/deploys/show.html.erb +8 -0
- data/app/views/jobs/show.html.erb +0 -34
- data/app/views/layouts/application.html.erb +1 -0
- data/app/views/layouts/minimal.html.erb +1 -0
- data/app/views/projects/_form.html.erb +0 -7
- data/app/views/releases/_form.html.erb +9 -3
- data/app/views/releases/index.html.erb +1 -1
- data/config/application.rb +1 -2
- data/config/database.yml +3 -3
- data/config/environments/production.rb +1 -1
- data/config/environments/test.rb +1 -1
- data/config/initializers/action_view_field_error_proc.rb +3 -3
- data/config/initializers/constants.rb +3 -3
- data/config/initializers/devise.rb +8 -8
- data/config/initializers/hard_coded_knowledge.rb +3 -3
- data/config/initializers/houston_async.rb +2 -2
- data/config/initializers/houston_report_exception.rb +3 -3
- data/config/initializers/houston_try.rb +3 -3
- data/config/initializers/sync_commits_on_post_receive.rb +0 -5
- data/config/routes.rb +108 -108
- data/db/migrate/20120417175450_devise_create_users.rb +2 -2
- data/db/migrate/20120501230243_add_unfuddle_id_to_tickets.rb +1 -1
- data/db/migrate/20120504143615_link_tickets_and_releases.rb +3 -3
- data/db/migrate/20120626150333_add_last_release_at_to_tickets.rb +2 -2
- data/db/migrate/20120626151320_link_tickets_and_commits.rb +2 -2
- data/db/migrate/20120626152020_add_committer_email_to_commits.rb +2 -2
- data/db/migrate/20120715230922_create_user_notifications.rb +3 -3
- data/db/migrate/20120716010743_add_user_to_releases.rb +3 -3
- data/db/migrate/20120726231754_link_projects_and_maintainers.rb +2 -2
- data/db/migrate/20120826022643_add_authentication_token_to_users.rb +2 -2
- data/db/migrate/20120920023251_add_administrator_to_users.rb +3 -3
- data/db/migrate/20121026014457_split_users_name_into_first_and_last.rb +2 -2
- data/db/migrate/20121027160548_create_deploys.rb +1 -1
- data/db/migrate/20121222170917_add_version_control_adapter_to_projects.rb +2 -2
- data/db/migrate/20121222223635_add_ticket_tracking_adapter_to_projects.rb +2 -2
- data/db/migrate/20121224212623_rename_environments.rb +3 -3
- data/db/migrate/20121225175106_replace_environment_id_with_environment_name.rb +8 -8
- data/db/migrate/20121230174234_create_test_runs.rb +3 -3
- data/db/migrate/20130119203853_remove_cached_queries_from_projects.rb +1 -1
- data/db/migrate/20130119204608_add_extended_attributes_to_tickets.rb +10 -10
- data/db/migrate/20130120182026_create_ticket_prerequisites.rb +2 -2
- data/db/migrate/20130312224911_create_errors.rb +1 -1
- data/db/migrate/20130407195450_create_roles.rb +2 -2
- data/db/migrate/20130407200624_populate_roles.rb +9 -9
- data/db/migrate/20130407220039_add_project_id_to_testing_notes.rb +6 -6
- data/db/migrate/20130407220937_add_project_id_to_commits.rb +6 -6
- data/db/migrate/20130416020627_add_environments_subscribed_to_to_users.rb +3 -3
- data/db/migrate/20130420174126_add_error_tracker_name_to_projects.rb +2 -2
- data/db/migrate/20130427223925_create_project_quotas.rb +2 -2
- data/db/migrate/20130504014802_add_extended_attributes_to_projects.rb +3 -3
- data/db/migrate/20130504135741_add_antecedents_to_tickets.rb +6 -6
- data/db/migrate/20130505144446_add_tags_to_tickets.rb +1 -1
- data/db/migrate/20130505162039_add_type_to_tickets.rb +1 -1
- data/db/migrate/20130505212838_add_closed_at_to_tickets.rb +1 -1
- data/db/migrate/20130519163615_create_user_credentials.rb +2 -2
- data/db/migrate/20130526024851_give_default_values_for_counts_from_test_runs.rb +2 -2
- data/db/migrate/20130710233849_add_authored_at_to_commits.rb +10 -10
- data/db/migrate/20130711013156_link_commits_and_users.rb +6 -6
- data/db/migrate/20130806143651_link_commits_and_releases.rb +2 -2
- data/db/migrate/20130921141449_create_milestones.rb +3 -3
- data/db/migrate/20131004185618_add_first_release_at_to_tickets.rb +2 -2
- data/db/migrate/20131012152403_create_sprints.rb +3 -3
- data/db/migrate/20131112010815_delete_duplicate_commits.rb +7 -7
- data/db/migrate/20140217160450_add_email_addresses_to_users.rb +3 -3
- data/db/migrate/20140327020121_drop_ticket_prerequisites.rb +1 -1
- data/db/migrate/20140406183224_remove_new_relic_id_from_projects.rb +1 -1
- data/db/migrate/20140406230121_delete_automatically_generated_messages_for_releases.rb +1 -1
- data/db/migrate/20140407010111_drop_sprints_project_id.rb +1 -1
- data/db/migrate/20140418133005_change_ticket_antecedents_to_text_array.rb +1 -1
- data/db/migrate/20140425141946_embed_changes_in_releases.rb +3 -3
- data/db/migrate/20140515174322_create_tasks.rb +3 -3
- data/db/migrate/20140516012049_create_tasks_for_existing_tickets.rb +1 -1
- data/db/migrate/20140517012626_prevent_hstore_fields_from_being_null.rb +2 -2
- data/db/migrate/20140521014652_allow_task_descriptions_to_be_null.rb +1 -1
- data/db/migrate/20140526155845_add_commit_id_to_deploys.rb +5 -5
- data/db/migrate/20140526162645_add_commit_before_id_and_commit_after_id_to_releases.rb +4 -4
- data/db/migrate/20140526180608_associate_remaining_deploys_and_releases.rb +12 -12
- data/db/migrate/20140724231918_create_sprints_tasks.rb +3 -3
- data/db/migrate/20140824194031_add_project_id_to_tasks.rb +1 -1
- data/db/migrate/20140824194526_add_completed_at_to_tasks.rb +3 -3
- data/db/migrate/20140824211249_require_sprints_to_be_unique.rb +1 -1
- data/db/migrate/20140907012329_drop_table_changes.rb +1 -1
- data/db/migrate/20140907013836_drop_table_errors.rb +1 -1
- data/db/migrate/20140921190022_add_checked_out_to_sprints_tasks.rb +2 -2
- data/db/migrate/20140921201441_drop_users_name.rb +1 -1
- data/db/migrate/20140925021043_rename_deploys_commit_to_sha.rb +1 -1
- data/db/migrate/20140929004347_create_historical_heads.rb +4 -4
- data/db/migrate/20141027194819_make_users_environments_subscribed_to_an_array.rb +2 -2
- data/db/migrate/20141226171730_create_measurements.rb +1 -1
- data/db/migrate/20150119154013_drop_unused_fields_from_milestones.rb +1 -1
- data/db/migrate/20150222205616_add_features_to_projects.rb +2 -2
- data/db/migrate/20150323004452_add_deploys_completed_at.rb +1 -1
- data/db/migrate/20150805180939_add_commit_id_to_test_runs.rb +1 -1
- data/db/migrate/20150808161805_create_test_results.rb +1 -1
- data/db/migrate/20150809132417_rename_test_results_regression_to_difference.rb +1 -1
- data/db/migrate/20150927014445_required_pull_requests_to_be_unique_by_project_and_number.rb +13 -0
- data/db/structure.sql +9 -0
- data/houston.gemspec +23 -26
- data/lib/configuration.rb +0 -5
- data/lib/core_ext/duration.rb +4 -4
- data/lib/core_ext/hash.rb +17 -17
- data/lib/houston/version.rb +1 -1
- data/lib/houston_server.rb +6 -6
- data/lib/parallel_enumerable.rb +16 -16
- data/lib/tasks/specific_tests.rake +37 -0
- metadata +91 -107
- data/app/interactors/push_gemfile_to_gemnasium.rb +0 -88
- data/app/models/gemnasium/alert.rb +0 -25
- data/app/views/welcome/_tdl.html.erb +0 -31
- data/config/initializers/houston_tdl.rb +0 -104
data/app/models/ticket_tag.rb
CHANGED
@@ -1,25 +1,25 @@
|
|
1
1
|
class TicketTag
|
2
|
-
|
2
|
+
|
3
3
|
def initialize(name, color)
|
4
4
|
@name = name
|
5
5
|
@color = color
|
6
6
|
end
|
7
|
-
|
7
|
+
|
8
8
|
attr_reader :name, :color
|
9
|
-
|
9
|
+
|
10
10
|
def self.from_s(string)
|
11
11
|
name, color = string.scan(/\[([^\]]+)\]\(([a-fA-F0-9]{6})\)/).flatten
|
12
12
|
name = string unless name
|
13
13
|
color = "e4e4e4" unless color
|
14
14
|
new(name, color)
|
15
15
|
end
|
16
|
-
|
16
|
+
|
17
17
|
def to_s
|
18
18
|
"[#{name}](#{color})"
|
19
19
|
end
|
20
|
-
|
20
|
+
|
21
21
|
def to_h
|
22
22
|
{name: name, color: color}
|
23
23
|
end
|
24
|
-
|
24
|
+
|
25
25
|
end
|
data/app/models/user.rb
CHANGED
@@ -1,6 +1,6 @@
|
|
1
1
|
class User < ActiveRecord::Base
|
2
2
|
include Retirement
|
3
|
-
|
3
|
+
|
4
4
|
has_many :testing_notes
|
5
5
|
has_many :roles, :dependent => :destroy
|
6
6
|
has_many :credentials, :class_name => "UserCredentials", :dependent => :destroy
|
@@ -8,123 +8,123 @@ class User < ActiveRecord::Base
|
|
8
8
|
has_many :tickets, foreign_key: "reporter_id"
|
9
9
|
has_and_belongs_to_many :commits
|
10
10
|
belongs_to :current_project, class_name: "Project"
|
11
|
-
|
11
|
+
|
12
12
|
devise *Houston.config.devise_configuration
|
13
|
-
|
13
|
+
|
14
14
|
default_scope { order("last_name, first_name") }
|
15
|
-
|
15
|
+
|
16
16
|
default_value_for :role, Houston.config.default_role
|
17
|
-
|
17
|
+
|
18
18
|
validates :first_name, :last_name, :email, :presence => true, :length => {:minimum => 2}
|
19
19
|
validates :role, presence: true, inclusion: {in: Houston.config.roles, message: "%{value} is not one of the configured roles (#{Houston.config.roles.join(", ")})"}
|
20
20
|
validate :all_email_addresses_must_be_unique
|
21
|
-
|
22
|
-
|
23
|
-
|
21
|
+
|
22
|
+
|
23
|
+
|
24
24
|
Houston.config.project_roles.each do |role|
|
25
25
|
method_name = role.downcase.gsub(' ', '_')
|
26
26
|
collection_name = method_name.pluralize
|
27
|
-
|
27
|
+
|
28
28
|
class_eval <<-RUBY
|
29
29
|
def self.#{collection_name}
|
30
30
|
Role.#{collection_name}.to_users
|
31
31
|
end
|
32
32
|
RUBY
|
33
33
|
end
|
34
|
-
|
34
|
+
|
35
35
|
Houston.config.roles.each do |role|
|
36
36
|
method_name = role.downcase.gsub(' ', '_')
|
37
37
|
collection_name = method_name.pluralize
|
38
|
-
|
38
|
+
|
39
39
|
class_eval <<-RUBY
|
40
40
|
def self.#{collection_name}
|
41
41
|
where(role: "#{role}")
|
42
42
|
end
|
43
|
-
|
43
|
+
|
44
44
|
def #{method_name}?
|
45
45
|
role == "#{role}"
|
46
46
|
end
|
47
47
|
RUBY
|
48
48
|
end
|
49
|
-
|
50
|
-
|
49
|
+
|
50
|
+
|
51
51
|
def self.administrators
|
52
52
|
where(administrator: true)
|
53
53
|
end
|
54
|
-
|
55
|
-
|
54
|
+
|
55
|
+
|
56
56
|
def self.participants
|
57
57
|
Role.participants.to_users
|
58
58
|
end
|
59
|
-
|
59
|
+
|
60
60
|
def self.with_primary_email(email)
|
61
61
|
email = email.downcase if email
|
62
62
|
where(email: email)
|
63
63
|
end
|
64
|
-
|
64
|
+
|
65
65
|
def self.with_email_address(*email_addresses)
|
66
66
|
email_addresses = email_addresses.flatten.compact
|
67
67
|
return none if email_addresses.none?
|
68
68
|
values = email_addresses.map { |email| connection.quote(email.downcase) }.join(",")
|
69
69
|
where("ARRAY[\"email_addresses\"] && ARRAY[#{values}]")
|
70
70
|
end
|
71
|
-
|
71
|
+
|
72
72
|
def self.find_by_email_address(email_address)
|
73
73
|
with_email_address(email_address).first
|
74
74
|
end
|
75
|
-
|
75
|
+
|
76
76
|
def self.with_view_option(option, value)
|
77
77
|
where(["view_options->? = ?", option, value])
|
78
78
|
end
|
79
|
-
|
80
|
-
|
81
|
-
|
79
|
+
|
80
|
+
|
81
|
+
|
82
82
|
def email=(value)
|
83
83
|
value = value.downcase if value
|
84
84
|
super(value)
|
85
85
|
self.email_addresses = [email] + alias_emails
|
86
86
|
end
|
87
|
-
|
87
|
+
|
88
88
|
def email_addresses
|
89
89
|
super || []
|
90
90
|
end
|
91
|
-
|
91
|
+
|
92
92
|
def alias_emails
|
93
93
|
email_addresses - [email]
|
94
94
|
end
|
95
|
-
|
95
|
+
|
96
96
|
def alias_emails=(value)
|
97
97
|
self.email_addresses = [email] + Array.wrap(value)
|
98
98
|
end
|
99
|
-
|
100
|
-
|
101
|
-
|
99
|
+
|
100
|
+
|
101
|
+
|
102
102
|
def name
|
103
103
|
"#{first_name} #{last_name}"
|
104
104
|
end
|
105
|
-
|
105
|
+
|
106
106
|
def follows?(project)
|
107
107
|
roles.to_projects.member?(project)
|
108
108
|
end
|
109
|
-
|
109
|
+
|
110
110
|
def followed_projects
|
111
111
|
roles.to_projects.unretired
|
112
112
|
end
|
113
|
-
|
113
|
+
|
114
114
|
def view_options
|
115
115
|
super || {}
|
116
116
|
end
|
117
|
-
|
118
|
-
|
119
|
-
|
117
|
+
|
118
|
+
|
119
|
+
|
120
120
|
# LDAP Overrides
|
121
121
|
# ------------------------------------------------------------------------- #
|
122
|
-
|
122
|
+
|
123
123
|
def self.find_ldap_entry(ldap_connection, auth_key_value)
|
124
124
|
filter = Net::LDAP::Filter.eq(Houston::TMI::FIELD_USED_FOR_LDAP_LOGIN, auth_key_value)
|
125
125
|
ldap_connection.ldap.search(filter: filter).first
|
126
126
|
end
|
127
|
-
|
127
|
+
|
128
128
|
def self.find_for_ldap_authentication(attributes, entry)
|
129
129
|
email = entry.mail.first.downcase
|
130
130
|
user = where(email: email).first
|
@@ -133,7 +133,7 @@ class User < ActiveRecord::Base
|
|
133
133
|
end
|
134
134
|
user
|
135
135
|
end
|
136
|
-
|
136
|
+
|
137
137
|
def self.create_from_ldap_entry(attributes, entry)
|
138
138
|
create!(
|
139
139
|
email: entry.mail.first.downcase,
|
@@ -142,13 +142,13 @@ class User < ActiveRecord::Base
|
|
142
142
|
first_name: entry.givenname.first,
|
143
143
|
last_name: entry.sn.first )
|
144
144
|
end
|
145
|
-
|
145
|
+
|
146
146
|
# ------------------------------------------------------------------------- #
|
147
|
-
|
148
|
-
|
149
|
-
|
147
|
+
|
148
|
+
|
149
|
+
|
150
150
|
private
|
151
|
-
|
151
|
+
|
152
152
|
def all_email_addresses_must_be_unique
|
153
153
|
email_addresses.each do |email_address|
|
154
154
|
if User.where(User.arel_table[:id].not_eq(id)).with_email_address(email_address).any?
|
@@ -156,5 +156,5 @@ private
|
|
156
156
|
end
|
157
157
|
end
|
158
158
|
end
|
159
|
-
|
159
|
+
|
160
160
|
end
|
@@ -1,16 +1,16 @@
|
|
1
1
|
class UserCredentials < ActiveRecord::Base
|
2
|
-
|
2
|
+
|
3
3
|
class MissingCredentials < RuntimeError; end
|
4
4
|
class InsufficientPermissions < RuntimeError; end
|
5
5
|
class InvalidCredentials < RuntimeError; end
|
6
6
|
class AccountLocked < RuntimeError; end
|
7
|
-
|
7
|
+
|
8
8
|
encrypt_with_public_key :password, key_pair: Houston.config.keypair
|
9
|
-
|
9
|
+
|
10
10
|
validate :test_connection
|
11
|
-
|
11
|
+
|
12
12
|
belongs_to :user
|
13
|
-
|
13
|
+
|
14
14
|
def self.for(service)
|
15
15
|
credentials = where(service: service).first || (raise MissingCredentials)
|
16
16
|
[credentials.login, credentials.password.decrypt(Houston.config.passphrase)]
|
@@ -18,9 +18,9 @@ class UserCredentials < ActiveRecord::Base
|
|
18
18
|
credentials.delete
|
19
19
|
raise MissingCredentials
|
20
20
|
end
|
21
|
-
|
22
|
-
|
23
|
-
|
21
|
+
|
22
|
+
|
23
|
+
|
24
24
|
def test_connection
|
25
25
|
case service
|
26
26
|
when "Unfuddle" then test_unfuddle_connection
|
@@ -28,7 +28,7 @@ class UserCredentials < ActiveRecord::Base
|
|
28
28
|
else raise NotImplementedError, "The service #{service.inspect} is not recognized"
|
29
29
|
end
|
30
30
|
end
|
31
|
-
|
31
|
+
|
32
32
|
def test_github_connection
|
33
33
|
password = self.password.decrypt(Houston.config.passphrase)
|
34
34
|
Octokit::Client.new(login: login, password: password).user
|
@@ -37,12 +37,12 @@ class UserCredentials < ActiveRecord::Base
|
|
37
37
|
rescue Octokit::Unauthorized
|
38
38
|
errors.add(:base, "Invalid credentials")
|
39
39
|
end
|
40
|
-
|
40
|
+
|
41
41
|
def test_unfuddle_connection
|
42
42
|
password = self.password.decrypt(Houston.config.passphrase)
|
43
43
|
Unfuddle.with_config(username: login, password: password) { Unfuddle.instance.get("people/current") }
|
44
44
|
rescue Unfuddle::UnauthorizedError
|
45
45
|
errors.add(:base, "Invalid credentials")
|
46
46
|
end
|
47
|
-
|
47
|
+
|
48
48
|
end
|
@@ -1,14 +1,14 @@
|
|
1
1
|
class CommitPresenter
|
2
2
|
include UrlHelper
|
3
|
-
|
3
|
+
|
4
4
|
def initialize(commits)
|
5
5
|
@commits = OneOrMany.new(commits)
|
6
6
|
end
|
7
|
-
|
7
|
+
|
8
8
|
def as_json(*args)
|
9
9
|
@commits.map(&method(:commit_to_json))
|
10
10
|
end
|
11
|
-
|
11
|
+
|
12
12
|
def commit_to_json(commit)
|
13
13
|
hash = {
|
14
14
|
id: commit.id,
|
@@ -19,7 +19,7 @@ class CommitPresenter
|
|
19
19
|
committer: {
|
20
20
|
name: commit.committer,
|
21
21
|
email: commit.committer_email } }
|
22
|
-
|
22
|
+
|
23
23
|
release = commit.releases.earliest
|
24
24
|
if release
|
25
25
|
# NB: we want to sort these with TesterNotes
|
@@ -31,7 +31,7 @@ class CommitPresenter
|
|
31
31
|
end
|
32
32
|
hash
|
33
33
|
end
|
34
|
-
|
34
|
+
|
35
35
|
def verbose
|
36
36
|
@commits.map do |commit|
|
37
37
|
commit_to_json(commit).merge({
|
@@ -48,5 +48,5 @@ class CommitPresenter
|
|
48
48
|
name: committer.name } } })
|
49
49
|
end
|
50
50
|
end
|
51
|
-
|
51
|
+
|
52
52
|
end
|
@@ -1,7 +1,7 @@
|
|
1
1
|
class OneOrMany < SimpleDelegator
|
2
|
-
|
2
|
+
|
3
3
|
delegate :is_a?, to: :__getobj__
|
4
|
-
|
4
|
+
|
5
5
|
def map(&block)
|
6
6
|
if __getobj__.respond_to?(:map)
|
7
7
|
__getobj__.map(&block)
|
@@ -9,7 +9,7 @@ class OneOrMany < SimpleDelegator
|
|
9
9
|
yield __getobj__
|
10
10
|
end
|
11
11
|
end
|
12
|
-
|
12
|
+
|
13
13
|
def select(&block)
|
14
14
|
if __getobj__.respond_to?(:select)
|
15
15
|
__getobj__.select(&block)
|
@@ -18,5 +18,5 @@ class OneOrMany < SimpleDelegator
|
|
18
18
|
self
|
19
19
|
end
|
20
20
|
end
|
21
|
-
|
21
|
+
|
22
22
|
end
|
@@ -1,20 +1,20 @@
|
|
1
1
|
class ProblemPresenter
|
2
2
|
include ActionView::Helpers::DateHelper
|
3
3
|
include CommitHelper
|
4
|
-
|
4
|
+
|
5
5
|
attr_reader :project, :problems
|
6
|
-
|
6
|
+
|
7
7
|
def initialize(project, problems)
|
8
8
|
@project = project
|
9
9
|
@problems = problems
|
10
10
|
end
|
11
|
-
|
11
|
+
|
12
12
|
def as_json(*args)
|
13
13
|
Houston.benchmark "[problem_presenter] Prepare JSON" do
|
14
14
|
@problems.map(&method(:problem_to_json))
|
15
15
|
end
|
16
16
|
end
|
17
|
-
|
17
|
+
|
18
18
|
def problem_to_json(problem)
|
19
19
|
{ id: problem.id,
|
20
20
|
ticketId: problem.ticket.try(:id),
|
@@ -26,11 +26,11 @@ class ProblemPresenter
|
|
26
26
|
where: problem.where,
|
27
27
|
noticesCount: problem.notices_count,
|
28
28
|
comments: problem.comments,
|
29
|
-
|
29
|
+
|
30
30
|
firstNotice: present_notice(problem.first_notice_at, problem.first_notice_commit, problem.first_notice_environment),
|
31
31
|
lastNotice: present_notice( problem.last_notice_at, problem.last_notice_commit, problem.last_notice_environment) }
|
32
32
|
end
|
33
|
-
|
33
|
+
|
34
34
|
def present_notice(time, sha, environment_name)
|
35
35
|
{ at: time,
|
36
36
|
ago: distance_of_time_in_words(time, Time.now).gsub("about ", "") + " ago",
|
@@ -38,12 +38,12 @@ class ProblemPresenter
|
|
38
38
|
commitUrl: (project.repo.commit_url(sha) if project.repo.respond_to?(:commit_url)),
|
39
39
|
release: present_release(sha, environment_name) }
|
40
40
|
end
|
41
|
-
|
41
|
+
|
42
42
|
def present_release(sha, environment_name)
|
43
43
|
release = @project.releases.where(["LOWER(environment_name) = ?", environment_name.downcase]).find_by_commit1(sha) if environment_name && !sha.blank?
|
44
44
|
{ url: "/projects/#{@project.slug}/environments/#{environment_name}/releases/#{release.id}",
|
45
45
|
at: release.created_at.strftime("%b %d"),
|
46
46
|
environment: environment_name.humanize } if release
|
47
47
|
end
|
48
|
-
|
48
|
+
|
49
49
|
end
|
@@ -1,9 +1,9 @@
|
|
1
1
|
class ReleasePresenter
|
2
|
-
|
2
|
+
|
3
3
|
def initialize(releases)
|
4
4
|
@releases = releases
|
5
5
|
end
|
6
|
-
|
6
|
+
|
7
7
|
def as_json(*args)
|
8
8
|
if @releases.is_a?(Release)
|
9
9
|
to_hash @releases
|
@@ -11,10 +11,10 @@ class ReleasePresenter
|
|
11
11
|
@releases.map(&method(:to_hash))
|
12
12
|
end
|
13
13
|
end
|
14
|
-
|
14
|
+
|
15
15
|
def to_hash(release)
|
16
16
|
{ id: release.id,
|
17
17
|
createdAt: release.created_at }
|
18
18
|
end
|
19
|
-
|
19
|
+
|
20
20
|
end
|
@@ -1,12 +1,12 @@
|
|
1
1
|
class SprintTaskPresenter < TaskPresenter
|
2
2
|
attr_reader :sprint, :ends_at
|
3
|
-
|
3
|
+
|
4
4
|
def initialize(sprint, tasks=sprint.tasks)
|
5
5
|
@sprint = sprint
|
6
6
|
@ends_at = sprint.end_date.end_of_day
|
7
7
|
super tasks
|
8
8
|
end
|
9
|
-
|
9
|
+
|
10
10
|
def task_to_json(task)
|
11
11
|
super.merge(
|
12
12
|
released: task.first_release_at && task.first_release_at < ends_at,
|
@@ -15,28 +15,28 @@ class SprintTaskPresenter < TaskPresenter
|
|
15
15
|
checkedOutAt: checked_out_at(task),
|
16
16
|
checkedOutBy: checked_out_by(task))
|
17
17
|
end
|
18
|
-
|
18
|
+
|
19
19
|
private
|
20
|
-
|
20
|
+
|
21
21
|
def checked_out_at(task)
|
22
22
|
checked_out(task)[:at]
|
23
23
|
end
|
24
|
-
|
24
|
+
|
25
25
|
def checked_out_by(task)
|
26
26
|
user_id = checked_out(task)[:by]
|
27
27
|
users[user_id] if user_id
|
28
28
|
end
|
29
|
-
|
29
|
+
|
30
30
|
def checked_out(task)
|
31
31
|
locks.fetch(task.id, {})
|
32
32
|
end
|
33
|
-
|
33
|
+
|
34
34
|
def locks
|
35
35
|
@locks ||= Hash[SprintTask.where(sprint_id: sprint.id, task_id: tasks.map(&:id))
|
36
36
|
.pluck(:task_id, :checked_out_at, :checked_out_by_id)
|
37
37
|
.map { |task_id, at, id| [task_id, {at: at, by: id}] }]
|
38
38
|
end
|
39
|
-
|
39
|
+
|
40
40
|
def users
|
41
41
|
@users ||= Hash[User.where(id: locks.values.map { |attrs| attrs[:by] })
|
42
42
|
.pluck(:id, :email, :first_name)
|
@@ -45,5 +45,5 @@ private
|
|
45
45
|
email: email,
|
46
46
|
firstName: first_name }] }]
|
47
47
|
end
|
48
|
-
|
48
|
+
|
49
49
|
end
|
@@ -1,12 +1,12 @@
|
|
1
1
|
class TaskPresenter
|
2
2
|
include UrlHelper
|
3
|
-
|
3
|
+
|
4
4
|
attr_reader :tasks
|
5
|
-
|
5
|
+
|
6
6
|
def initialize(tasks)
|
7
7
|
@tasks = OneOrMany.new(tasks)
|
8
8
|
end
|
9
|
-
|
9
|
+
|
10
10
|
def as_json(*args)
|
11
11
|
tasks = @tasks
|
12
12
|
tasks = Houston.benchmark "[#{self.class.name.underscore}] Load objects" do
|
@@ -16,23 +16,23 @@ class TaskPresenter
|
|
16
16
|
tasks.select(&:ticket).map(&method(:task_to_json))
|
17
17
|
end
|
18
18
|
end
|
19
|
-
|
19
|
+
|
20
20
|
def task_to_json(task)
|
21
21
|
ticket = task.ticket
|
22
22
|
project = ticket.project
|
23
23
|
{ id: task.id,
|
24
|
-
|
24
|
+
|
25
25
|
projectId: project.id,
|
26
26
|
projectSlug: project.slug,
|
27
27
|
projectTitle: project.name,
|
28
28
|
projectColor: project.color,
|
29
|
-
|
29
|
+
|
30
30
|
ticketSystem: project.ticket_tracker_name,
|
31
31
|
ticketUrl: ticket.ticket_tracker_ticket_url,
|
32
32
|
ticketNumber: ticket.number,
|
33
33
|
ticketType: ticket.type.to_s.downcase.dasherize,
|
34
34
|
ticketSequence: ticket.extended_attributes["sequence"], # <-- embeds knowledge of Houston::Scheduler
|
35
|
-
|
35
|
+
|
36
36
|
shorthand: task.shorthand,
|
37
37
|
description: task.description,
|
38
38
|
effort: task.effort,
|
@@ -40,5 +40,5 @@ class TaskPresenter
|
|
40
40
|
firstCommitAt: task.first_commit_at,
|
41
41
|
completedAt: task.completed_at }
|
42
42
|
end
|
43
|
-
|
43
|
+
|
44
44
|
end
|
@@ -85,23 +85,23 @@ class TasksExcelPresenter
|
|
85
85
|
package.to_stream.string
|
86
86
|
end
|
87
87
|
end
|
88
|
-
|
88
|
+
|
89
89
|
private
|
90
|
-
|
90
|
+
|
91
91
|
def commit_time(commits)
|
92
92
|
return nil if commits.none?
|
93
93
|
commits.sum { |_, message, _| Commit.parse_message(message)[:hours_worked] }
|
94
94
|
end
|
95
|
-
|
95
|
+
|
96
96
|
def committers(commits)
|
97
97
|
commits.flat_map { |_, _, committer| committer.split(" and ") }.uniq
|
98
98
|
end
|
99
|
-
|
99
|
+
|
100
100
|
def commits_by_task_id
|
101
101
|
@commits_by_task_id ||= Commit.joins(:tasks).merge(tasks)
|
102
102
|
.pluck("tasks.id", :message, :committer)
|
103
103
|
.group_by { |id, _, _| id }
|
104
104
|
end
|
105
|
-
|
106
|
-
|
105
|
+
|
106
|
+
|
107
107
|
end
|
@@ -1,10 +1,10 @@
|
|
1
1
|
class TestingNotePresenter
|
2
2
|
include AvatarHelper
|
3
|
-
|
3
|
+
|
4
4
|
def initialize(testing_notes)
|
5
5
|
@testing_notes = testing_notes
|
6
6
|
end
|
7
|
-
|
7
|
+
|
8
8
|
def as_json(*args)
|
9
9
|
if @testing_notes.is_a?(TestingNote)
|
10
10
|
to_hash @testing_notes
|
@@ -12,7 +12,7 @@ class TestingNotePresenter
|
|
12
12
|
@testing_notes.map(&method(:to_hash))
|
13
13
|
end
|
14
14
|
end
|
15
|
-
|
15
|
+
|
16
16
|
def to_hash(testing_note)
|
17
17
|
{ id: testing_note.id,
|
18
18
|
createdAt: testing_note.created_at,
|
@@ -23,5 +23,5 @@ class TestingNotePresenter
|
|
23
23
|
verdict: testing_note.verdict,
|
24
24
|
comment: testing_note.comment }
|
25
25
|
end
|
26
|
-
|
26
|
+
|
27
27
|
end
|