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/concerns/nosync.rb
CHANGED
@@ -1,5 +1,5 @@
|
|
1
1
|
module Nosync
|
2
|
-
|
2
|
+
|
3
3
|
def nosync
|
4
4
|
value = nosync?
|
5
5
|
begin
|
@@ -9,13 +9,13 @@ module Nosync
|
|
9
9
|
self.nosync = value
|
10
10
|
end
|
11
11
|
end
|
12
|
-
|
12
|
+
|
13
13
|
def nosync=(value)
|
14
14
|
@nosync = value
|
15
15
|
end
|
16
|
-
|
16
|
+
|
17
17
|
def nosync?
|
18
18
|
!!@nosync
|
19
19
|
end
|
20
|
-
|
20
|
+
|
21
21
|
end
|
@@ -1,81 +1,81 @@
|
|
1
1
|
module ProjectAdapter
|
2
|
-
|
3
|
-
|
2
|
+
|
3
|
+
|
4
4
|
def adapters
|
5
5
|
@adapters ||= {}
|
6
6
|
end
|
7
|
-
|
8
|
-
|
7
|
+
|
8
|
+
|
9
9
|
def has_adapter(*adapter_namespaces)
|
10
10
|
adapter_namespaces.each do |adapter_namespace|
|
11
11
|
adapter_module = Houston::Adapters[adapter_namespace]
|
12
12
|
raise ArgumentError, "#{adapter_module} should respond to `adapters`" unless adapter_module.respond_to?(:adapters)
|
13
13
|
raise ArgumentError, "#{adapter_module} should respond to `adapter`" unless adapter_module.respond_to?(:adapter)
|
14
|
-
|
14
|
+
|
15
15
|
adapter = Adapter.new(self, adapter_module)
|
16
16
|
adapters[adapter.name] = adapter
|
17
|
-
|
17
|
+
|
18
18
|
adapter.define_methods!
|
19
|
-
|
19
|
+
|
20
20
|
validate adapter.validation_method
|
21
21
|
end
|
22
22
|
end
|
23
|
-
|
24
|
-
|
23
|
+
|
24
|
+
|
25
25
|
class Adapter
|
26
|
-
|
26
|
+
|
27
27
|
def initialize(model, adapter_module)
|
28
28
|
@model = model
|
29
29
|
@namespace = adapter_module
|
30
30
|
@name = adapter_module.name
|
31
31
|
@attribute_name = name.demodulize.underscore
|
32
32
|
end
|
33
|
-
|
33
|
+
|
34
34
|
attr_reader :model, :namespace, :name, :attribute_name
|
35
|
-
|
35
|
+
|
36
36
|
def title
|
37
37
|
name.demodulize.titleize
|
38
38
|
end
|
39
|
-
|
39
|
+
|
40
40
|
def validation_method
|
41
41
|
:"#{attribute_name}_configuration_is_valid"
|
42
42
|
end
|
43
|
-
|
43
|
+
|
44
44
|
def define_methods!
|
45
45
|
model.module_eval <<-RUBY
|
46
46
|
def self.with_#{attribute_name}
|
47
47
|
where arel_table[:#{attribute_name}_name].not_eq("None")
|
48
48
|
end
|
49
|
-
|
49
|
+
|
50
50
|
def has_#{attribute_name}?
|
51
51
|
#{attribute_name}_name != "None"
|
52
52
|
end
|
53
|
-
|
53
|
+
|
54
54
|
def #{validation_method}
|
55
55
|
#{attribute_name}_adapter.errors_with_parameters(self, *parameters_for_#{attribute_name}_adapter.values).each do |attribute, messages|
|
56
56
|
errors.add(attribute, messages) if messages.any?
|
57
57
|
end
|
58
58
|
end
|
59
|
-
|
59
|
+
|
60
60
|
def #{attribute_name}
|
61
61
|
@#{attribute_name} ||= #{attribute_name}_adapter
|
62
62
|
.build(self, *parameters_for_#{attribute_name}_adapter.values)
|
63
63
|
.extend(FeatureSupport)
|
64
64
|
end
|
65
|
-
|
65
|
+
|
66
66
|
def parameters_for_#{attribute_name}_adapter
|
67
67
|
#{attribute_name}_adapter.parameters.each_with_object({}) do |parameter, hash|
|
68
68
|
hash[parameter] = extended_attributes[parameter.to_s]
|
69
69
|
end
|
70
70
|
end
|
71
|
-
|
71
|
+
|
72
72
|
def #{attribute_name}_adapter
|
73
73
|
#{namespace}.adapter(#{attribute_name}_name)
|
74
74
|
end
|
75
75
|
RUBY
|
76
76
|
end
|
77
|
-
|
77
|
+
|
78
78
|
end
|
79
|
-
|
80
|
-
|
79
|
+
|
80
|
+
|
81
81
|
end
|
data/app/concerns/retirement.rb
CHANGED
@@ -1,28 +1,28 @@
|
|
1
1
|
module Retirement
|
2
2
|
extend ActiveSupport::Concern
|
3
|
-
|
3
|
+
|
4
4
|
module ClassMethods
|
5
5
|
def unretired
|
6
6
|
where(retired_at: nil)
|
7
7
|
end
|
8
|
-
|
8
|
+
|
9
9
|
def retired
|
10
10
|
where(arel_table[:retired_at].not_eq(nil))
|
11
11
|
end
|
12
12
|
end
|
13
|
-
|
14
|
-
|
13
|
+
|
14
|
+
|
15
15
|
def retire!
|
16
16
|
update_column(:retired_at, Time.now)
|
17
17
|
freeze
|
18
18
|
end
|
19
|
-
|
19
|
+
|
20
20
|
def unretire!
|
21
21
|
update_column(:retired_at, nil)
|
22
22
|
end
|
23
|
-
|
23
|
+
|
24
24
|
def retired?
|
25
25
|
retired_at.present?
|
26
26
|
end
|
27
|
-
|
27
|
+
|
28
28
|
end
|
@@ -1,79 +1,79 @@
|
|
1
1
|
module TicketSynchronizer
|
2
|
-
|
3
|
-
|
2
|
+
|
3
|
+
|
4
4
|
def fetch_all
|
5
5
|
return all if !ticket_tracker.supports?(:syncing_tickets)
|
6
|
-
|
6
|
+
|
7
7
|
Houston.benchmark "GET All Tickets" do
|
8
8
|
synchronize ticket_tracker.all_tickets
|
9
9
|
end
|
10
10
|
end
|
11
|
-
|
11
|
+
|
12
12
|
def fetch_open
|
13
13
|
return open if !ticket_tracker.supports?(:syncing_tickets)
|
14
|
-
|
14
|
+
|
15
15
|
Houston.benchmark "GET Open Tickets" do
|
16
16
|
synchronize ticket_tracker.open_tickets
|
17
17
|
end
|
18
18
|
end
|
19
|
-
|
19
|
+
|
20
20
|
def find_by_number!(number)
|
21
21
|
numbered(number, sync: true).first || (raise ActiveRecord::RecordNotFound)
|
22
22
|
end
|
23
|
-
|
23
|
+
|
24
24
|
def numbered(*numbers, sync: false)
|
25
25
|
numbers = numbers.flatten.map(&:to_i).uniq
|
26
26
|
return none if numbers.empty?
|
27
|
-
|
27
|
+
|
28
28
|
results = super(*numbers)
|
29
29
|
return results unless sync && ticket_tracker.supports?(:syncing_tickets)
|
30
|
-
|
30
|
+
|
31
31
|
results.concat fetch_numbered(numbers - results.map(&:number))
|
32
32
|
end
|
33
|
-
|
33
|
+
|
34
34
|
def fetch_numbered(numbers)
|
35
35
|
return [] if numbers.empty?
|
36
36
|
return numbered(numbers) if !ticket_tracker.supports?(:syncing_tickets)
|
37
|
-
|
37
|
+
|
38
38
|
Houston.benchmark "GET Numbered Tickets" do
|
39
39
|
synchronize ticket_tracker.find_tickets_numbered(numbers)
|
40
40
|
end
|
41
41
|
end
|
42
|
-
|
42
|
+
|
43
43
|
def fetch_with_query(*query)
|
44
44
|
return [] unless ticket_tracker.respond_to?(:find_tickets!) # <-- an optional API
|
45
|
-
|
45
|
+
|
46
46
|
Rails.logger.info "[tickets.fetch_with_query] query: #{query.inspect}"
|
47
|
-
|
47
|
+
|
48
48
|
synchronize ticket_tracker.find_tickets!(*query)
|
49
49
|
end
|
50
|
-
|
51
|
-
|
50
|
+
|
51
|
+
|
52
52
|
def synchronize(unsynchronized_tickets)
|
53
53
|
unsynchronized_tickets = unsynchronized_tickets.reject(&:nil?)
|
54
54
|
return [] if unsynchronized_tickets.empty?
|
55
|
-
|
55
|
+
|
56
56
|
map_milestone_id = project.milestones.remote_id_map
|
57
|
-
|
57
|
+
|
58
58
|
Houston.benchmark("[tickets.synchronize] synchronizing #{unsynchronized_tickets.length} tickets") do
|
59
59
|
numbers = unsynchronized_tickets.map(&:number)
|
60
60
|
tickets = Ticket.unscoped { where(number: numbers) }
|
61
|
-
|
61
|
+
|
62
62
|
unsynchronized_tickets.map do |unsynchronized_ticket|
|
63
63
|
ticket = tickets.detect { |ticket| ticket.number == unsynchronized_ticket.number }
|
64
64
|
attributes = unsynchronized_ticket.attributes.merge(destroyed_at: nil)
|
65
|
-
|
65
|
+
|
66
66
|
# Convert remote milestone IDs to local milestone IDs
|
67
67
|
attributes[:milestone_id] = map_milestone_id[attributes[:milestone_id]]
|
68
|
-
|
68
|
+
|
69
69
|
if ticket
|
70
|
-
|
70
|
+
|
71
71
|
# This is essentially a call to update_attributes,
|
72
72
|
# but I broke it down so that we don't begin a
|
73
73
|
# transaction if we don't have any changes to save.
|
74
74
|
# This is pretty much just to reduce log verbosity.
|
75
75
|
ticket.assign_attributes(attributes)
|
76
|
-
|
76
|
+
|
77
77
|
# hstore always thinks it has changed
|
78
78
|
has_legitimate_changes = ticket.changed?
|
79
79
|
if has_legitimate_changes && ticket.changed == %w{extended_attributes}
|
@@ -87,7 +87,7 @@ module TicketSynchronizer
|
|
87
87
|
else
|
88
88
|
ticket = Ticket.nosync { create(attributes) }
|
89
89
|
end
|
90
|
-
|
90
|
+
|
91
91
|
# There's no reason why this shouldn't be set,
|
92
92
|
# but in order to reduce a bunch of useless hits
|
93
93
|
# to the cache and a bunch of log output...
|
@@ -96,8 +96,8 @@ module TicketSynchronizer
|
|
96
96
|
end
|
97
97
|
end
|
98
98
|
end
|
99
|
-
|
100
|
-
|
99
|
+
|
100
|
+
|
101
101
|
def create_from_remote(remote_ticket)
|
102
102
|
attributes = remote_ticket.attributes
|
103
103
|
if project.ticket_tracker.features.include?(:syncing_milestones)
|
@@ -105,16 +105,16 @@ module TicketSynchronizer
|
|
105
105
|
end
|
106
106
|
create(attributes)
|
107
107
|
end
|
108
|
-
|
109
|
-
|
108
|
+
|
109
|
+
|
110
110
|
private
|
111
|
-
|
111
|
+
|
112
112
|
def ticket_tracker
|
113
113
|
project.ticket_tracker
|
114
114
|
end
|
115
|
-
|
115
|
+
|
116
116
|
def project
|
117
117
|
proxy_association.owner
|
118
118
|
end
|
119
|
-
|
119
|
+
|
120
120
|
end
|
data/app/concerns/unique_add.rb
CHANGED
@@ -5,44 +5,44 @@ module Api
|
|
5
5
|
before_filter :find_sprint
|
6
6
|
before_filter :find_task, only: [:create, :destroy]
|
7
7
|
skip_before_filter :verify_authenticity_token
|
8
|
-
|
8
|
+
|
9
9
|
attr_reader :sprint, :task
|
10
|
-
|
10
|
+
|
11
11
|
rescue_from ActiveRecord::RecordNotFound do
|
12
12
|
head 404
|
13
13
|
end
|
14
|
-
|
14
|
+
|
15
15
|
def index
|
16
16
|
render json: sprint.tasks
|
17
17
|
.includes(:ticket => :project)
|
18
18
|
.map { |task| present_task(task) }
|
19
19
|
end
|
20
|
-
|
20
|
+
|
21
21
|
def mine
|
22
22
|
render json: sprint.tasks
|
23
23
|
.includes(:ticket => :project)
|
24
24
|
.checked_out_by(current_user)
|
25
25
|
.map { |task| present_task(task) }
|
26
26
|
end
|
27
|
-
|
27
|
+
|
28
28
|
def create
|
29
29
|
authorize! :update, sprint
|
30
|
-
|
30
|
+
|
31
31
|
if sprint.completed?
|
32
32
|
render text: "The Sprint is completed. You cannot add or remove tasks.", status: :unprocessable_entity
|
33
33
|
return
|
34
34
|
end
|
35
|
-
|
35
|
+
|
36
36
|
if sprint.locked? && !task.ticket_id.in?(sprint.ticket_ids)
|
37
37
|
render text: "The Sprint is locked. You can add tasks for tickets that are already in the Sprint, but you can't add new tickets to the Sprint.", status: :unprocessable_entity
|
38
38
|
return
|
39
39
|
end
|
40
|
-
|
40
|
+
|
41
41
|
# Putting a task into a Sprint implies that you're able to estimate this ticket
|
42
42
|
task.ticket.able_to_estimate! if task.ticket.respond_to?(:able_to_estimate!)
|
43
|
-
|
43
|
+
|
44
44
|
task.update_attributes(effort: params[:effort]) if params[:effort]
|
45
|
-
|
45
|
+
|
46
46
|
if task.completed? && task.completed_at < sprint.starts_at
|
47
47
|
render text: "Task ##{task.shorthand} cannot be added to the Sprint because it was completed before the Sprint began", status: :unprocessable_entity
|
48
48
|
elsif task.effort.nil? or task.effort.zero?
|
@@ -53,26 +53,26 @@ module Api
|
|
53
53
|
render json: SprintTaskPresenter.new(sprint, task).to_json
|
54
54
|
end
|
55
55
|
end
|
56
|
-
|
56
|
+
|
57
57
|
def destroy
|
58
58
|
authorize! :update, sprint
|
59
|
-
|
59
|
+
|
60
60
|
if sprint.completed?
|
61
61
|
render text: "The Sprint is completed. You cannot add or remove tasks.", status: :unprocessable_entity
|
62
62
|
return
|
63
63
|
end
|
64
|
-
|
64
|
+
|
65
65
|
if sprint.locked?
|
66
66
|
render text: "The Sprint is locked; tasks cannot be removed", status: :unprocessable_entity
|
67
67
|
return
|
68
68
|
end
|
69
|
-
|
69
|
+
|
70
70
|
SprintTask.where(sprint_id: sprint.id, task_id: task.id).delete_all
|
71
71
|
head :ok
|
72
72
|
end
|
73
|
-
|
73
|
+
|
74
74
|
private
|
75
|
-
|
75
|
+
|
76
76
|
def present_task(task)
|
77
77
|
{ projectSlug: task.project.slug,
|
78
78
|
number: task.number,
|
@@ -83,15 +83,15 @@ module Api
|
|
83
83
|
releasedAt: task.first_release_at,
|
84
84
|
completedAt: task.completed_at }
|
85
85
|
end
|
86
|
-
|
86
|
+
|
87
87
|
def find_sprint
|
88
88
|
@sprint = Sprint.current || Sprint.new
|
89
89
|
end
|
90
|
-
|
90
|
+
|
91
91
|
def find_task
|
92
92
|
@task = Task.find_by_project_and_shorthand(params[:project_slug], params[:shorthand]) || (raise ActiveRecord::RecordNotFound)
|
93
93
|
end
|
94
|
-
|
94
|
+
|
95
95
|
end
|
96
96
|
end
|
97
97
|
end
|
@@ -5,22 +5,22 @@ module Api
|
|
5
5
|
before_filter :find_project_and_ticket
|
6
6
|
before_filter :find_task, only: [:update, :destroy]
|
7
7
|
skip_before_filter :verify_authenticity_token
|
8
|
-
|
8
|
+
|
9
9
|
attr_reader :project, :ticket, :task
|
10
|
-
|
10
|
+
|
11
11
|
rescue_from ActiveRecord::RecordNotFound do
|
12
12
|
head 404
|
13
13
|
end
|
14
|
-
|
14
|
+
|
15
15
|
def index
|
16
16
|
authorize! :read, Task
|
17
17
|
render json: ticket.tasks.map { |task| present_task(task) }
|
18
18
|
end
|
19
|
-
|
19
|
+
|
20
20
|
def create
|
21
21
|
task = ticket.tasks.build params.slice(:description, :effort)
|
22
22
|
authorize! :create, task
|
23
|
-
|
23
|
+
|
24
24
|
task.updated_by = current_user
|
25
25
|
if task.save
|
26
26
|
render json: present_task(task), status: :created
|
@@ -28,10 +28,10 @@ module Api
|
|
28
28
|
render json: {errors: task.errors.full_messages}, status: :unprocessable_entity
|
29
29
|
end
|
30
30
|
end
|
31
|
-
|
31
|
+
|
32
32
|
def update
|
33
33
|
authorize! :update, task
|
34
|
-
|
34
|
+
|
35
35
|
task.attributes = params.slice(:description, :effort)
|
36
36
|
task.updated_by = current_user
|
37
37
|
if task.save
|
@@ -40,16 +40,16 @@ module Api
|
|
40
40
|
render json: {errors: task.errors.full_messages}, status: :unprocessable_entity
|
41
41
|
end
|
42
42
|
end
|
43
|
-
|
43
|
+
|
44
44
|
def destroy
|
45
45
|
authorize! :destroy, task
|
46
|
-
|
46
|
+
|
47
47
|
task.destroy
|
48
48
|
head :ok
|
49
49
|
end
|
50
|
-
|
50
|
+
|
51
51
|
private
|
52
|
-
|
52
|
+
|
53
53
|
def present_task(task)
|
54
54
|
{ id: task.id,
|
55
55
|
number: task.number,
|
@@ -60,16 +60,16 @@ module Api
|
|
60
60
|
releasedAt: task.first_release_at,
|
61
61
|
completedAt: task.completed_at }
|
62
62
|
end
|
63
|
-
|
63
|
+
|
64
64
|
def find_project_and_ticket
|
65
65
|
@project = Project.find_by_slug! params[:slug]
|
66
66
|
@ticket = project.tickets.find_by_number! params[:number]
|
67
67
|
end
|
68
|
-
|
68
|
+
|
69
69
|
def find_task
|
70
70
|
@task = ticket.tasks.find params[:id]
|
71
71
|
end
|
72
|
-
|
72
|
+
|
73
73
|
end
|
74
74
|
end
|
75
75
|
end
|
@@ -1,15 +1,15 @@
|
|
1
1
|
class ApplicationController < ActionController::Base
|
2
2
|
include UrlHelper
|
3
|
-
|
3
|
+
|
4
4
|
# Prevent CSRF attacks by raising an exception.
|
5
5
|
# For APIs, you may want to use :null_session instead.
|
6
6
|
protect_from_forgery with: :exception
|
7
|
-
|
7
|
+
|
8
8
|
before_filter :set_current_project
|
9
9
|
after_filter :save_current_project
|
10
|
-
|
11
|
-
|
12
|
-
|
10
|
+
|
11
|
+
|
12
|
+
|
13
13
|
rescue_from CanCan::AccessDenied do |exception|
|
14
14
|
if current_user
|
15
15
|
if request.xhr?
|
@@ -23,19 +23,19 @@ class ApplicationController < ActionController::Base
|
|
23
23
|
require_login
|
24
24
|
end
|
25
25
|
end
|
26
|
-
|
26
|
+
|
27
27
|
rescue_from UserCredentials::MissingCredentials do
|
28
28
|
head 401, "X-Credentials" => "Missing Credentials"
|
29
29
|
end
|
30
|
-
|
30
|
+
|
31
31
|
rescue_from UserCredentials::InvalidCredentials do
|
32
32
|
head 401, "X-Credentials" => "Invalid Credentials"
|
33
33
|
end
|
34
|
-
|
34
|
+
|
35
35
|
rescue_from UserCredentials::InsufficientPermissions do |exception|
|
36
36
|
render text: exception.message, status: 401
|
37
37
|
end
|
38
|
-
|
38
|
+
|
39
39
|
rescue_from Github::Unauthorized do |exception|
|
40
40
|
session["user.return_to"] = request.referer
|
41
41
|
if request.xhr?
|
@@ -44,44 +44,50 @@ class ApplicationController < ActionController::Base
|
|
44
44
|
redirect_to oauth_consumer_path(id: "github")
|
45
45
|
end
|
46
46
|
end
|
47
|
-
|
47
|
+
|
48
48
|
rescue_from ActiveRecord::RecordNotFound do
|
49
49
|
render file: "public/404", layout: false
|
50
50
|
end
|
51
|
-
|
51
|
+
|
52
52
|
# Malformed request
|
53
53
|
rescue_from ActionController::UnknownFormat do
|
54
54
|
head 400
|
55
55
|
end unless Rails.env.development?
|
56
|
-
|
57
|
-
|
58
|
-
|
56
|
+
|
57
|
+
|
58
|
+
|
59
59
|
def require_login
|
60
60
|
redirect_to main_app.new_user_session_path
|
61
61
|
end
|
62
|
-
|
63
|
-
|
64
|
-
|
62
|
+
|
63
|
+
|
64
|
+
|
65
|
+
def unfurling?
|
66
|
+
request.env["HTTP_USER_AGENT"] =~ /^Slackbot-LinkExpanding/
|
67
|
+
end
|
68
|
+
|
69
|
+
|
70
|
+
|
65
71
|
def after_sign_in_path_for(user)
|
66
72
|
path = session["user_redirect_to"] || stored_location_for(user) || root_path
|
67
73
|
path = root_path if path =~ /\/users\/(sign_in|password)/
|
68
74
|
path
|
69
75
|
end
|
70
|
-
|
71
|
-
|
72
|
-
|
76
|
+
|
77
|
+
|
78
|
+
|
73
79
|
def revision
|
74
80
|
expire_revision!
|
75
81
|
return_or_cache_revision!
|
76
82
|
end
|
77
|
-
|
83
|
+
|
78
84
|
def expire_revision!
|
79
85
|
if session[:revision_expiration].blank? || session[:revision_expiration] < Time.now.utc
|
80
86
|
session[:revision] = nil
|
81
87
|
Rails.logger.info "[revision] expiring"
|
82
88
|
end
|
83
89
|
end
|
84
|
-
|
90
|
+
|
85
91
|
def return_or_cache_revision!
|
86
92
|
session[:revision] || read_revision.tap do |sha|
|
87
93
|
session[:revision] = sha
|
@@ -89,17 +95,17 @@ class ApplicationController < ActionController::Base
|
|
89
95
|
Rails.logger.info "[revision] sha: #{sha[0..8]}, expiration: #{session[:revision_expiration]}"
|
90
96
|
end
|
91
97
|
end
|
92
|
-
|
98
|
+
|
93
99
|
def read_revision
|
94
100
|
revision_path = Rails.root.join("REVISION")
|
95
101
|
File.exists?(revision_path) ? File.read(revision_path).chomp : ""
|
96
102
|
end
|
97
|
-
|
98
|
-
|
99
|
-
|
103
|
+
|
104
|
+
|
105
|
+
|
100
106
|
def api_authenticate!
|
101
107
|
return if current_user
|
102
|
-
|
108
|
+
|
103
109
|
allow_params_authentication!
|
104
110
|
authenticate_or_request_with_http_basic do |username, password|
|
105
111
|
params["user"] ||= {}
|
@@ -112,40 +118,40 @@ class ApplicationController < ActionController::Base
|
|
112
118
|
end
|
113
119
|
end
|
114
120
|
end
|
115
|
-
|
116
|
-
|
117
|
-
|
121
|
+
|
122
|
+
|
123
|
+
|
118
124
|
helper_method :followed_projects
|
119
125
|
def followed_projects
|
120
126
|
return @followed_projects if defined?(@followed_projects)
|
121
127
|
return @followed_projects = [] unless current_user
|
122
128
|
@followed_projects = current_user.followed_projects.to_a
|
123
129
|
end
|
124
|
-
|
130
|
+
|
125
131
|
helper_method :current_project
|
126
132
|
def current_project
|
127
133
|
@current_project ||= @project || (@default_project_slug ? Project[@default_project_slug] : (current_user && current_user.current_project))
|
128
134
|
end
|
129
|
-
|
135
|
+
|
130
136
|
def set_current_project
|
131
137
|
@default_project_slug = params[:project] if params[:project].is_a?(String)
|
132
138
|
end
|
133
|
-
|
139
|
+
|
134
140
|
def save_current_project
|
135
141
|
return unless current_user && current_project
|
136
|
-
|
142
|
+
|
137
143
|
current_user.current_project_id = current_project.id
|
138
144
|
current_user.save if current_user.current_project_id_changed?
|
139
145
|
end
|
140
|
-
|
141
|
-
|
142
|
-
|
146
|
+
|
147
|
+
|
148
|
+
|
143
149
|
def no_cache
|
144
150
|
response.headers["Cache-Control"] = "no-cache, no-store, max-age=0, must-revalidate"
|
145
151
|
response.headers["Pragma"] = "no-cache"
|
146
152
|
response.headers["Expires"] = "Fri, 01 Jan 1990 00:00:00 GMT"
|
147
153
|
end
|
148
|
-
|
149
|
-
|
150
|
-
|
154
|
+
|
155
|
+
|
156
|
+
|
151
157
|
end
|