foreman_patch 1.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/LICENSE +619 -0
- data/README.md +38 -0
- data/Rakefile +53 -0
- data/app/assets/javascript/foreman_patch/plan_edit_windows.js +9 -0
- data/app/assets/stylesheets/foreman_patch/cycle_plans.scss +6 -0
- data/app/assets/stylesheets/foreman_patch/foreman_patch.css +4 -0
- data/app/controllers/concerns/foreman_patch/parameters/ticket_field.rb +26 -0
- data/app/controllers/foreman_patch/api/v2/base_controller.rb +12 -0
- data/app/controllers/foreman_patch/api/v2/cycles_controller.rb +78 -0
- data/app/controllers/foreman_patch/api/v2/groups_controller.rb +70 -0
- data/app/controllers/foreman_patch/api/v2/host_groups_controller.rb +13 -0
- data/app/controllers/foreman_patch/api/v2/invocations_controller.rb +70 -0
- data/app/controllers/foreman_patch/api/v2/plans_controller.rb +79 -0
- data/app/controllers/foreman_patch/api/v2/rounds_controller.rb +77 -0
- data/app/controllers/foreman_patch/api/v2/window_plans_controller.rb +73 -0
- data/app/controllers/foreman_patch/api/v2/windows_controller.rb +85 -0
- data/app/controllers/foreman_patch/concerns/api/v2/hosts_controller_extensions.rb +21 -0
- data/app/controllers/foreman_patch/concerns/hosts_controller_extensions.rb +61 -0
- data/app/controllers/foreman_patch/cycles_controller.rb +34 -0
- data/app/controllers/foreman_patch/groups_controller.rb +68 -0
- data/app/controllers/foreman_patch/invocations_controller.rb +51 -0
- data/app/controllers/foreman_patch/plans_controller.rb +59 -0
- data/app/controllers/foreman_patch/rounds_controller.rb +18 -0
- data/app/controllers/foreman_patch/ticket_fields_controller.rb +37 -0
- data/app/controllers/foreman_patch/window_plans_controller.rb +60 -0
- data/app/controllers/foreman_patch/windows_controller.rb +58 -0
- data/app/helpers/concerns/foreman_patch/hosts_helper_extensions.rb +13 -0
- data/app/helpers/foreman_patch/cycles_helper.rb +19 -0
- data/app/helpers/foreman_patch/hosts_helper.rb +52 -0
- data/app/helpers/foreman_patch/patching_helper.rb +116 -0
- data/app/helpers/foreman_patch/plans_helper.rb +45 -0
- data/app/helpers/foreman_patch/ticket_helper.rb +11 -0
- data/app/helpers/foreman_patch/window_patching_helper.rb +20 -0
- data/app/helpers/foreman_patch/window_plans_helper.rb +13 -0
- data/app/helpers/foreman_patch/windows_helper.rb +17 -0
- data/app/lib/actions/foreman_patch/cycle/complete.rb +41 -0
- data/app/lib/actions/foreman_patch/cycle/create.rb +69 -0
- data/app/lib/actions/foreman_patch/cycle/initiate.rb +68 -0
- data/app/lib/actions/foreman_patch/cycle/plan.rb +73 -0
- data/app/lib/actions/foreman_patch/cycle/prepare_content.rb +123 -0
- data/app/lib/actions/foreman_patch/host/reschedule.rb +32 -0
- data/app/lib/actions/foreman_patch/invocation/action.rb +135 -0
- data/app/lib/actions/foreman_patch/invocation/ensure_services.rb +47 -0
- data/app/lib/actions/foreman_patch/invocation/patch.rb +93 -0
- data/app/lib/actions/foreman_patch/invocation/reschedule.rb +21 -0
- data/app/lib/actions/foreman_patch/invocation/restart.rb +101 -0
- data/app/lib/actions/foreman_patch/invocation/update_packages.rb +52 -0
- data/app/lib/actions/foreman_patch/invocation/wait_for_host.rb +103 -0
- data/app/lib/actions/foreman_patch/round/add_missing_hosts.rb +21 -0
- data/app/lib/actions/foreman_patch/round/create.rb +37 -0
- data/app/lib/actions/foreman_patch/round/patch.rb +81 -0
- data/app/lib/actions/foreman_patch/round/plan.rb +33 -0
- data/app/lib/actions/foreman_patch/round/resolve_hosts.rb +45 -0
- data/app/lib/actions/foreman_patch/window/create.rb +41 -0
- data/app/lib/actions/foreman_patch/window/patch.rb +72 -0
- data/app/lib/actions/foreman_patch/window/plan.rb +43 -0
- data/app/lib/actions/foreman_patch/window/publish.rb +32 -0
- data/app/lib/actions/foreman_patch/window/resolve_hosts.rb +31 -0
- data/app/lib/actions/helpers/failure_notification.rb +20 -0
- data/app/lib/actions/helpers/with_feature_action.rb +102 -0
- data/app/lib/actions/middleware/check_exit_status.rb +31 -0
- data/app/mailers/foreman_patch/cycle_mailer.rb +17 -0
- data/app/mailers/foreman_patch/group_mailer.rb +41 -0
- data/app/mailers/foreman_patch/invocation_mailer.rb +19 -0
- data/app/models/foreman_patch/concerns/group_facet_host_extensions.rb +35 -0
- data/app/models/foreman_patch/concerns/host_managed_extensions.rb +17 -0
- data/app/models/foreman_patch/cycle.rb +49 -0
- data/app/models/foreman_patch/group.rb +43 -0
- data/app/models/foreman_patch/host/group_facet.rb +13 -0
- data/app/models/foreman_patch/invocation.rb +60 -0
- data/app/models/foreman_patch/plan.rb +110 -0
- data/app/models/foreman_patch/plan_task_group.rb +11 -0
- data/app/models/foreman_patch/round.rb +68 -0
- data/app/models/foreman_patch/ticket_field.rb +23 -0
- data/app/models/foreman_patch/window.rb +123 -0
- data/app/models/foreman_patch/window_plan.rb +77 -0
- data/app/models/foreman_patch/window_task_group.rb +11 -0
- data/app/models/setting/patching.rb +57 -0
- data/app/services/foreman_patch/cycle_name_generator.rb +38 -0
- data/app/services/foreman_patch/ticket/affected_items.rb +110 -0
- data/app/services/foreman_patch/ticket/api.rb +68 -0
- data/app/services/foreman_patch/ticket/change_request.rb +92 -0
- data/app/services/foreman_patch/ticket/field_render.rb +21 -0
- data/app/services/foreman_patch/ticket/payload.rb +136 -0
- data/app/services/foreman_patch/ticket.rb +14 -0
- data/app/views/dashboard/_foreman_patch_widget.html.erb +2 -0
- data/app/views/foreman_patch/api/v2/cycles/base.json.rabl +3 -0
- data/app/views/foreman_patch/api/v2/cycles/create.json.rabl +3 -0
- data/app/views/foreman_patch/api/v2/cycles/index.json.rabl +3 -0
- data/app/views/foreman_patch/api/v2/cycles/show.json.rabl +9 -0
- data/app/views/foreman_patch/api/v2/cycles/update.json.rabl +3 -0
- data/app/views/foreman_patch/api/v2/group_facet/base.json.rabl +7 -0
- data/app/views/foreman_patch/api/v2/group_facet/base_with_root.json.rabl +4 -0
- data/app/views/foreman_patch/api/v2/group_facet/show.json.rabl +3 -0
- data/app/views/foreman_patch/api/v2/groups/base.json.rabl +3 -0
- data/app/views/foreman_patch/api/v2/groups/create.json.rabl +3 -0
- data/app/views/foreman_patch/api/v2/groups/index.json.rabl +2 -0
- data/app/views/foreman_patch/api/v2/groups/show.json.rabl +14 -0
- data/app/views/foreman_patch/api/v2/groups/update.json.rabl +3 -0
- data/app/views/foreman_patch/api/v2/invocations/base.json.rabl +6 -0
- data/app/views/foreman_patch/api/v2/invocations/index.json.rabl +3 -0
- data/app/views/foreman_patch/api/v2/invocations/phase.json.rabl +7 -0
- data/app/views/foreman_patch/api/v2/invocations/show.json.rabl +7 -0
- data/app/views/foreman_patch/api/v2/plans/base.json.rabl +3 -0
- data/app/views/foreman_patch/api/v2/plans/create.json.rabl +2 -0
- data/app/views/foreman_patch/api/v2/plans/index.json.rabl +3 -0
- data/app/views/foreman_patch/api/v2/plans/show.json.rabl +9 -0
- data/app/views/foreman_patch/api/v2/plans/update.json.rabl +2 -0
- data/app/views/foreman_patch/api/v2/rounds/base.json.rabl +3 -0
- data/app/views/foreman_patch/api/v2/rounds/index.json.rabl +3 -0
- data/app/views/foreman_patch/api/v2/rounds/show.json.rabl +17 -0
- data/app/views/foreman_patch/api/v2/rounds/status.json.rabl +11 -0
- data/app/views/foreman_patch/api/v2/window_plans/base.json.rabl +4 -0
- data/app/views/foreman_patch/api/v2/window_plans/create.json.rabl +2 -0
- data/app/views/foreman_patch/api/v2/window_plans/index.json.rabl +2 -0
- data/app/views/foreman_patch/api/v2/window_plans/show.json.rabl +14 -0
- data/app/views/foreman_patch/api/v2/window_plans/update.json.rabl +2 -0
- data/app/views/foreman_patch/api/v2/windows/base.json.rabl +3 -0
- data/app/views/foreman_patch/api/v2/windows/index.json.rabl +3 -0
- data/app/views/foreman_patch/api/v2/windows/schedule.json.rabl +13 -0
- data/app/views/foreman_patch/api/v2/windows/show.json.rabl +13 -0
- data/app/views/foreman_patch/api/v2/windows/update.json.rabl +3 -0
- data/app/views/foreman_patch/cycle_mailer/_details.html.erb +4 -0
- data/app/views/foreman_patch/cycle_mailer/_round.html.erb +12 -0
- data/app/views/foreman_patch/cycle_mailer/_summary.html.erb +6 -0
- data/app/views/foreman_patch/cycle_mailer/_window.html.erb +4 -0
- data/app/views/foreman_patch/cycle_mailer/planned.html.erb +7 -0
- data/app/views/foreman_patch/cycles/_form.html.erb +9 -0
- data/app/views/foreman_patch/cycles/edit.html.erb +41 -0
- data/app/views/foreman_patch/cycles/index.html.erb +45 -0
- data/app/views/foreman_patch/cycles/show.html.erb +26 -0
- data/app/views/foreman_patch/group_mailer/_dashboard.html.erb +31 -0
- data/app/views/foreman_patch/group_mailer/_list.html.erb +17 -0
- data/app/views/foreman_patch/group_mailer/completed.html.erb +16 -0
- data/app/views/foreman_patch/group_mailer/initiated.html.erb +4 -0
- data/app/views/foreman_patch/groups/_form.html.erb +9 -0
- data/app/views/foreman_patch/groups/edit.html.erb +3 -0
- data/app/views/foreman_patch/groups/index.html.erb +40 -0
- data/app/views/foreman_patch/groups/new.html.erb +3 -0
- data/app/views/foreman_patch/hosts/hosts/new_action.html.erb +1 -0
- data/app/views/foreman_patch/hosts/new_action.html.erb +1 -0
- data/app/views/foreman_patch/invocation_mailer/_output_line_set.html.erb +6 -0
- data/app/views/foreman_patch/invocation_mailer/_output_line_set.text.erb +3 -0
- data/app/views/foreman_patch/invocation_mailer/failure.html.erb +50 -0
- data/app/views/foreman_patch/invocation_mailer/failure.text.erb +10 -0
- data/app/views/foreman_patch/invocations/_output_line_set.html.erb +8 -0
- data/app/views/foreman_patch/invocations/_phase.html.erb +17 -0
- data/app/views/foreman_patch/invocations/_primary.html.erb +1 -0
- data/app/views/foreman_patch/invocations/_refresh.js.erb +7 -0
- data/app/views/foreman_patch/invocations/show.html.erb +44 -0
- data/app/views/foreman_patch/invocations/show.js.erb +17 -0
- data/app/views/foreman_patch/layouts/react.html.erb +17 -0
- data/app/views/foreman_patch/plans/_form.html.erb +12 -0
- data/app/views/foreman_patch/plans/edit.html.erb +6 -0
- data/app/views/foreman_patch/plans/index.html.erb +32 -0
- data/app/views/foreman_patch/plans/new.html.erb +3 -0
- data/app/views/foreman_patch/plans/show.html.erb +19 -0
- data/app/views/foreman_patch/rounds/show.html.erb +54 -0
- data/app/views/foreman_patch/ticket_fields/_fields.html.erb +66 -0
- data/app/views/foreman_patch/ticket_fields/edit.html.erb +6 -0
- data/app/views/foreman_patch/ticket_fields/index.html.erb +26 -0
- data/app/views/foreman_patch/ticket_fields/new.html.erb +6 -0
- data/app/views/foreman_patch/window_plans/_form.html.erb +11 -0
- data/app/views/foreman_patch/window_plans/_hidden_layout.html.erb +7 -0
- data/app/views/foreman_patch/window_plans/edit.html.erb +3 -0
- data/app/views/foreman_patch/window_plans/new.html.erb +3 -0
- data/app/views/foreman_patch/windows/_form.html.erb +21 -0
- data/app/views/foreman_patch/windows/_groups.html.erb +2 -0
- data/app/views/foreman_patch/windows/_rounds.html.erb +24 -0
- data/app/views/foreman_patch/windows/_schedule.html.erb +16 -0
- data/app/views/foreman_patch/windows/_ticket.html.erb +12 -0
- data/app/views/foreman_patch/windows/edit.html.erb +3 -0
- data/app/views/foreman_patch/windows/new.html.erb +3 -0
- data/app/views/foreman_patch/windows/show.html.erb +27 -0
- data/app/views/foreman_patch/windows/show.json.erb +4 -0
- data/app/views/hosts/select_multiple_patch_group.html.erb +12 -0
- data/app/views/overrides/patch_groups/_host_patch_group_select.html.erb +9 -0
- data/app/views/templates/ensure_services.erb +28 -0
- data/config/api_routes.rb +37 -0
- data/config/initializers/pagelets.rb +6 -0
- data/config/initializers/safemode_jail.rb +4 -0
- data/config/routes/mount_engine.rb +3 -0
- data/config/routes/overrides.rb +10 -0
- data/config/routes.rb +47 -0
- data/db/migrate/20210202161304_create_foreman_patch_plans.rb +32 -0
- data/db/migrate/20210202163323_create_foreman_patch_cycles.rb +38 -0
- data/db/migrate/20210202164301_create_foreman_patch_groups.rb +37 -0
- data/db/migrate/20210226134103_add_name_to_cycles.rb +31 -0
- data/db/migrate/20210226162824_rename_default_window.rb +5 -0
- data/db/migrate/20210302165058_add_task_to_window_groups.rb +7 -0
- data/db/migrate/20210304141111_create_invocation.rb +16 -0
- data/db/migrate/20210519163923_add_cycle_end_date.rb +29 -0
- data/db/migrate/20210525154113_expand_window_groups.rb +27 -0
- data/db/migrate/20210723160142_add_cycle_plan_task_group.rb +7 -0
- data/db/migrate/20210831160044_cycle_plan_start_correction.rb +5 -0
- data/db/migrate/20210907104645_rename_group_priority.rb +5 -0
- data/db/migrate/20210909130118_remove_old_references.rb +14 -0
- data/db/migrate/20210910141428_rename_round.rb +7 -0
- data/db/migrate/20210910163542_rename_plan.rb +8 -0
- data/db/migrate/20211014212415_round_group_reference.rb +9 -0
- data/db/migrate/20220105224803_add_group_label.rb +25 -0
- data/db/migrate/20220114153808_plan_add_name_generator.rb +5 -0
- data/db/migrate/20220117103808_remove_cycle_description.rb +5 -0
- data/db/migrate/20220406110705_add_window_state.rb +37 -0
- data/db/migrate/20220407161120_add_round_status.rb +9 -0
- data/db/migrate/20220902134800_add_invocation_status.rb +28 -0
- data/db/seeds.d/100-assign_features_with_templates.rb +16 -0
- data/db/seeds.d/160-mail_notifications.rb +38 -0
- data/db/seeds.d/75-job_templates.rb +16 -0
- data/lib/foreman_patch/engine.rb +87 -0
- data/lib/foreman_patch/plugin.rb +47 -0
- data/lib/foreman_patch/version.rb +3 -0
- data/lib/foreman_patch.rb +4 -0
- data/lib/tasks/foreman_patch_tasks.rake +47 -0
- data/locale/Makefile +60 -0
- data/locale/en/foreman_patch.po +19 -0
- data/locale/foreman_patch.pot +19 -0
- data/locale/gemspec.rb +2 -0
- data/package.json +35 -0
- data/public/assets/foreman_patch/cycle_plans-e5667e178ba389908f5c815b24ec0ea77c340849d56bc39c5ce72bb626bd446a.scss +6 -0
- data/public/assets/foreman_patch/cycle_plans-e5667e178ba389908f5c815b24ec0ea77c340849d56bc39c5ce72bb626bd446a.scss.gz +0 -0
- data/public/assets/foreman_patch/foreman_patch-ce5805a60c0d5f896f557ff5246e5a09172043004c850b39bea54e618df1c485.css +1 -0
- data/public/assets/foreman_patch/foreman_patch-ce5805a60c0d5f896f557ff5246e5a09172043004c850b39bea54e618df1c485.css.gz +0 -0
- data/public/assets/foreman_patch/foreman_patch.json +1 -0
- data/public/assets/foreman_patch/plan_edit_windows-e656ba411642a7f983b51958ab30ac49c056322d19295a603cff4d5e6c71c8ed.js +1 -0
- data/public/assets/foreman_patch/plan_edit_windows-e656ba411642a7f983b51958ab30ac49c056322d19295a603cff4d5e6c71c8ed.js.gz +0 -0
- data/public/webpack/foreman_patch/bundle-200e97f4e2ad9ed413ea.css +1 -0
- data/public/webpack/foreman_patch/bundle-200e97f4e2ad9ed413ea.css.gz +0 -0
- data/public/webpack/foreman_patch/bundle-200e97f4e2ad9ed413ea.js +6 -0
- data/public/webpack/foreman_patch/bundle-200e97f4e2ad9ed413ea.js.gz +0 -0
- data/public/webpack/foreman_patch/bundle-200e97f4e2ad9ed413ea.js.map +1 -0
- data/public/webpack/foreman_patch/bundle-200e97f4e2ad9ed413ea.js.map.gz +0 -0
- data/public/webpack/foreman_patch/foreman_patch-1e4f2d5e6f040a27aa7a.css +1 -0
- data/public/webpack/foreman_patch/foreman_patch-1e4f2d5e6f040a27aa7a.css.gz +0 -0
- data/public/webpack/foreman_patch/foreman_patch-1e4f2d5e6f040a27aa7a.js +6 -0
- data/public/webpack/foreman_patch/foreman_patch-1e4f2d5e6f040a27aa7a.js.gz +0 -0
- data/public/webpack/foreman_patch/foreman_patch-1e4f2d5e6f040a27aa7a.js.map +1 -0
- data/public/webpack/foreman_patch/foreman_patch-1e4f2d5e6f040a27aa7a.js.map.gz +0 -0
- data/public/webpack/foreman_patch/manifest.json +26 -0
- data/public/webpack/foreman_patch/manifest.json.gz +0 -0
- data/public/webpack/foreman_patch/vendor-4b77c91f1e9103179596.js +2 -0
- data/public/webpack/foreman_patch/vendor-4b77c91f1e9103179596.js.gz +0 -0
- data/public/webpack/foreman_patch/vendor-4b77c91f1e9103179596.js.map +1 -0
- data/public/webpack/foreman_patch/vendor-4b77c91f1e9103179596.js.map.gz +0 -0
- data/test/factories/foreman_patch_factories.rb +5 -0
- data/test/test_plugin_helper.rb +6 -0
- data/test/unit/foreman_patch_test.rb +11 -0
- data/webpack/components/Cycle/Cycle.js +57 -0
- data/webpack/components/Cycle/CycleActions.js +21 -0
- data/webpack/components/Cycle/CycleConstants.js +2 -0
- data/webpack/components/Cycle/CycleHelpers.js +3 -0
- data/webpack/components/Cycle/CycleSelectors.js +10 -0
- data/webpack/components/Cycle/Window.js +32 -0
- data/webpack/components/Cycle/index.js +41 -0
- data/webpack/components/Groups/Group.js +20 -0
- data/webpack/components/Groups/Groups.js +46 -0
- data/webpack/components/Groups/GroupsHelpers.js +11 -0
- data/webpack/components/Groups/Priority.js +51 -0
- data/webpack/components/Invocation/Invocation.js +47 -0
- data/webpack/components/Invocation/InvocationActions.js +8 -0
- data/webpack/components/Invocation/InvocationConsts.js +1 -0
- data/webpack/components/Invocation/InvocationSelectors.js +14 -0
- data/webpack/components/Invocation/index.js +36 -0
- data/webpack/components/Invocations/Invocations.js +55 -0
- data/webpack/components/Invocations/InvocationsConstants.js +1 -0
- data/webpack/components/Invocations/InvocationsHelpers.js +11 -0
- data/webpack/components/Invocations/InvocationsPage.js +66 -0
- data/webpack/components/Invocations/InvocationsPage.scss +3 -0
- data/webpack/components/Invocations/InvocationsSelectors.js +21 -0
- data/webpack/components/Invocations/components/InvocationItem.js +66 -0
- data/webpack/components/Invocations/components/InvocationStatus.js +51 -0
- data/webpack/components/Invocations/index.js +99 -0
- data/webpack/components/Plan/Plan.js +90 -0
- data/webpack/components/Plan/PlanActions.js +12 -0
- data/webpack/components/Plan/PlanConstants.js +1 -0
- data/webpack/components/Plan/PlanSelectors.js +9 -0
- data/webpack/components/Plan/Window.js +33 -0
- data/webpack/components/Plan/index.js +36 -0
- data/webpack/components/PlanWindow/PlanWindow.js +55 -0
- data/webpack/components/PlanWindow/index.js +32 -0
- data/webpack/components/RoundProgress/AggregateStatus.js +58 -0
- data/webpack/components/RoundProgress/RoundProgress.js +48 -0
- data/webpack/components/RoundProgress/RoundProgressActions.js +9 -0
- data/webpack/components/RoundProgress/RoundProgressConstants.js +1 -0
- data/webpack/components/RoundProgress/RoundProgressSelectors.js +24 -0
- data/webpack/components/RoundProgress/index.js +44 -0
- data/webpack/components/Rounds/Rounds.js +57 -0
- data/webpack/components/Rounds/RoundsActions.js +8 -0
- data/webpack/components/Rounds/RoundsConsts.js +1 -0
- data/webpack/components/Rounds/RoundsSelectors.js +14 -0
- data/webpack/components/Rounds/components/RoundItem.js +52 -0
- data/webpack/components/Rounds/components/RoundStatus.js +40 -0
- data/webpack/components/Rounds/index.js +37 -0
- data/webpack/components/common/Calendar/Calendar.js +86 -0
- data/webpack/components/common/Calendar/Calendar.scss +76 -0
- data/webpack/components/common/Calendar/CalendarConstants.js +1 -0
- data/webpack/components/common/Calendar/CalendarHelpers.js +77 -0
- data/webpack/components/common/Calendar/Day/index.js +9 -0
- data/webpack/components/common/Calendar/Event.js +25 -0
- data/webpack/components/common/Calendar/Month/Day.js +51 -0
- data/webpack/components/common/Calendar/Month/Header.js +77 -0
- data/webpack/components/common/Calendar/Month/Month.js +47 -0
- data/webpack/components/common/Calendar/Month/index.js +1 -0
- data/webpack/components/common/Calendar/View.js +20 -0
- data/webpack/components/common/Calendar/Week/index.js +13 -0
- data/webpack/components/common/Calendar/index.js +2 -0
- data/webpack/components/common/Terminal/OutputLine.js +26 -0
- data/webpack/components/common/Terminal/Terminal.js +115 -0
- data/webpack/components/common/Terminal/Terminal.scss +47 -0
- data/webpack/index.js +20 -0
- metadata +438 -0
@@ -0,0 +1,14 @@
|
|
1
|
+
import {
|
2
|
+
selectAPIStatus,
|
3
|
+
selectAPIResponse,
|
4
|
+
} from 'foremanReact/redux/API/APISelectors';
|
5
|
+
import { ROUNDS } from './RoundsConsts';
|
6
|
+
|
7
|
+
export const selectItems = state =>
|
8
|
+
selectAPIResponse(state, ROUNDS).rounds || [];
|
9
|
+
|
10
|
+
export const selectAutoRefresh = state =>
|
11
|
+
selectAPIResponse(state, ROUNDS).autoRefresh;
|
12
|
+
|
13
|
+
export const selectStatus = state => selectAPIStatus(state, ROUNDS);
|
14
|
+
|
@@ -0,0 +1,52 @@
|
|
1
|
+
import React from 'react';
|
2
|
+
import PropTypes from 'prop-types';
|
3
|
+
import { ActionButtons } from 'foremanReact/components/common/ActionButtons/ActionButtons';
|
4
|
+
import RoundStatus from './RoundStatus';
|
5
|
+
|
6
|
+
const RoundItem = ({ name, link, priority, hostsCount, hostsLink, status, actions }) => {
|
7
|
+
const roundLink = link ? (
|
8
|
+
<a href={link}>{name}</a>
|
9
|
+
) : (
|
10
|
+
<a href="#" className="disabled">
|
11
|
+
{name}
|
12
|
+
</a>
|
13
|
+
);
|
14
|
+
|
15
|
+
const hostsCountLink = hostsLink ? (
|
16
|
+
<a href={hostsLink}>{hostsCount}</a>
|
17
|
+
) : (
|
18
|
+
<a href="#" className="disabled">
|
19
|
+
{hostsCount}
|
20
|
+
</a>
|
21
|
+
);
|
22
|
+
|
23
|
+
return (
|
24
|
+
<tr id={`round-${name}`}>
|
25
|
+
<td className="round_name">{roundLink}</td>
|
26
|
+
<td className="round_priority">{priority}</td>
|
27
|
+
<td className="round_hosts_count">{hostsCountLink}</td>
|
28
|
+
<td className="round_status">
|
29
|
+
<RoundStatus status={status} />
|
30
|
+
</td>
|
31
|
+
<td className="round_actions">
|
32
|
+
<ActionButtons buttons={[...actions]} />
|
33
|
+
</td>
|
34
|
+
</tr>
|
35
|
+
);
|
36
|
+
};
|
37
|
+
|
38
|
+
RoundItem.propTypes = {
|
39
|
+
name: PropTypes.string.isRequired,
|
40
|
+
link: PropTypes.string.isRequired,
|
41
|
+
priority: PropTypes.number.isRequired,
|
42
|
+
hostsCount: PropTypes.number.isRequired,
|
43
|
+
hostsLink: PropTypes.string.isRequired,
|
44
|
+
status: PropTypes.string.isRequired,
|
45
|
+
actions: PropTypes.array,
|
46
|
+
};
|
47
|
+
|
48
|
+
RoundItem.defaultProps = {
|
49
|
+
actions: [],
|
50
|
+
};
|
51
|
+
|
52
|
+
export default RoundItem;
|
@@ -0,0 +1,40 @@
|
|
1
|
+
import React from 'react';
|
2
|
+
import { Icon } from 'patternfly-react';
|
3
|
+
import PropTypes from 'prop-types';
|
4
|
+
import { translate as __ } from 'foremanReact/common/I18n';
|
5
|
+
|
6
|
+
const RoundStatus = ({ status }) => {
|
7
|
+
switch (status) {
|
8
|
+
case 'complete':
|
9
|
+
return (
|
10
|
+
<div>
|
11
|
+
<Icon type="pf" name="ok" /> {status}
|
12
|
+
</div>
|
13
|
+
);
|
14
|
+
case 'running':
|
15
|
+
return (
|
16
|
+
<div>
|
17
|
+
<Icon type="pf" name="in-progress" /> {status}
|
18
|
+
</div>
|
19
|
+
);
|
20
|
+
case 'pending':
|
21
|
+
case 'planned':
|
22
|
+
return (
|
23
|
+
<div>
|
24
|
+
<Icon type="pf" name="pending" /> {status}
|
25
|
+
</div>
|
26
|
+
);
|
27
|
+
default:
|
28
|
+
return (
|
29
|
+
<div>
|
30
|
+
<Icon type="fa" name="question" /> {status}
|
31
|
+
</div>
|
32
|
+
);
|
33
|
+
}
|
34
|
+
};
|
35
|
+
|
36
|
+
RoundStatus.propTypes = {
|
37
|
+
status: PropTypes.string.isRequired,
|
38
|
+
};
|
39
|
+
|
40
|
+
export default RoundStatus;
|
@@ -0,0 +1,37 @@
|
|
1
|
+
import React, { useEffect, useState } from 'react';
|
2
|
+
import { useSelector, useDispatch } from 'react-redux';
|
3
|
+
import { stopInterval } from 'foremanReact/redux/middlewares/IntervalMiddleware';
|
4
|
+
import Rounds from './Rounds';
|
5
|
+
|
6
|
+
import {
|
7
|
+
selectItems,
|
8
|
+
selectStatus,
|
9
|
+
selectAutoRefresh,
|
10
|
+
} from './RoundsSelectors';
|
11
|
+
import { getData } from './RoundsActions';
|
12
|
+
import { ROUNDS } from './RoundsConsts';
|
13
|
+
|
14
|
+
const WrappedRounds = () => {
|
15
|
+
const dispatch = useDispatch();
|
16
|
+
const autoRefresh = useSelector(selectAutoRefresh);
|
17
|
+
const items = useSelector(selectItems);
|
18
|
+
const status = useSelector(selectStatus);
|
19
|
+
|
20
|
+
useEffect(() => {
|
21
|
+
dispatch(getData());
|
22
|
+
|
23
|
+
return () => {
|
24
|
+
dispatch(stopInterval(ROUNDS));
|
25
|
+
};
|
26
|
+
}, [dispatch]);
|
27
|
+
|
28
|
+
useEffect(() => {
|
29
|
+
if (autoRefresh === 'false') {
|
30
|
+
dispatch(stopInterval(ROUNDS));
|
31
|
+
}
|
32
|
+
}, [autoRefresh, dispatch]);
|
33
|
+
|
34
|
+
return <Rounds status={status} items={items} />;
|
35
|
+
};
|
36
|
+
|
37
|
+
export default WrappedRounds;
|
@@ -0,0 +1,86 @@
|
|
1
|
+
import React, { useState } from 'react';
|
2
|
+
import PropTypes from 'prop-types';
|
3
|
+
import { DndProvider } from 'react-dnd';
|
4
|
+
import HTML5Backend from 'react-dnd-html5-backend';
|
5
|
+
import './Calendar.scss';
|
6
|
+
|
7
|
+
import { views, getView } from './View';
|
8
|
+
|
9
|
+
const Calendar = (props) => {
|
10
|
+
const {start, end, weekStartsOn, locale, onEventMoved} = props;
|
11
|
+
|
12
|
+
const initialDate = () => {
|
13
|
+
const { date } = props;
|
14
|
+
if (date < start || date > end) {
|
15
|
+
return (date < start) ? new Date(start) : new Date(end);
|
16
|
+
} else {
|
17
|
+
return date;
|
18
|
+
}
|
19
|
+
};
|
20
|
+
|
21
|
+
const [date, setDate] = useState(initialDate());
|
22
|
+
const [view, setView] = useState(props.view);
|
23
|
+
const [events, setEvents] = useState(props.events);
|
24
|
+
|
25
|
+
const movedCallback = (updatedEvent) => {
|
26
|
+
onEventMoved(updatedEvent);
|
27
|
+
|
28
|
+
const result = events.map(event => {
|
29
|
+
if (event.id === updatedEvent.id) {
|
30
|
+
return {...event, ...updatedEvent};
|
31
|
+
}
|
32
|
+
return event;
|
33
|
+
});
|
34
|
+
|
35
|
+
setEvents(result);
|
36
|
+
};
|
37
|
+
|
38
|
+
let View = getView(view);
|
39
|
+
|
40
|
+
return (
|
41
|
+
<div className="calendar">
|
42
|
+
<DndProvider backend={HTML5Backend}>
|
43
|
+
<View
|
44
|
+
start={start}
|
45
|
+
end={end}
|
46
|
+
date={date}
|
47
|
+
setDate={setDate}
|
48
|
+
setView={view}
|
49
|
+
events={events}
|
50
|
+
onEventMoved={movedCallback}
|
51
|
+
locale={locale}
|
52
|
+
weekStartsOn={weekStartsOn}
|
53
|
+
/>
|
54
|
+
</DndProvider>
|
55
|
+
</div>
|
56
|
+
);
|
57
|
+
};
|
58
|
+
|
59
|
+
Calendar.propTypes = {
|
60
|
+
start: PropTypes.instanceOf(Date),
|
61
|
+
end: PropTypes.instanceOf(Date),
|
62
|
+
date: PropTypes.instanceOf(Date),
|
63
|
+
view: PropTypes.oneOf(Object.values(views)),
|
64
|
+
events: PropTypes.arrayOf(PropTypes.shape({
|
65
|
+
id: PropTypes.number.isRequired,
|
66
|
+
start: PropTypes.instanceOf(Date).isRequired,
|
67
|
+
end: PropTypes.instanceOf(Date).isRequired,
|
68
|
+
title: PropTypes.node.isRequired,
|
69
|
+
})),
|
70
|
+
onEventMoved: PropTypes.func,
|
71
|
+
locale: PropTypes.string,
|
72
|
+
weekStartsOn: PropTypes.number,
|
73
|
+
};
|
74
|
+
|
75
|
+
Calendar.defaultProps = {
|
76
|
+
start: null,
|
77
|
+
end: null,
|
78
|
+
date: new Date(),
|
79
|
+
view: views.MONTH,
|
80
|
+
events: [],
|
81
|
+
onEventMoved: (event) => {},
|
82
|
+
locale: 'en-US',
|
83
|
+
weekStartsOn: 1,
|
84
|
+
};
|
85
|
+
|
86
|
+
export default Calendar;
|
@@ -0,0 +1,76 @@
|
|
1
|
+
.calendar {
|
2
|
+
|
3
|
+
.day {
|
4
|
+
padding: 2px;
|
5
|
+
height: 120px;
|
6
|
+
}
|
7
|
+
|
8
|
+
.wday-0 {}
|
9
|
+
.wday-1 {}
|
10
|
+
.wday-2 {}
|
11
|
+
.wday-3 {}
|
12
|
+
.wday-4 {}
|
13
|
+
.wday-5 {}
|
14
|
+
.wday-6 {}
|
15
|
+
|
16
|
+
.today {
|
17
|
+
background: #ffffc0;
|
18
|
+
}
|
19
|
+
|
20
|
+
.past {
|
21
|
+
background: #f9f9f9;
|
22
|
+
}
|
23
|
+
|
24
|
+
.future {}
|
25
|
+
|
26
|
+
.start-date {}
|
27
|
+
|
28
|
+
.disabled {
|
29
|
+
background: #e8e8e8;
|
30
|
+
}
|
31
|
+
.enabled {}
|
32
|
+
td.enabled:hover {
|
33
|
+
background: #def3ff;
|
34
|
+
}
|
35
|
+
|
36
|
+
h6 {
|
37
|
+
margin: 2px 3px;
|
38
|
+
}
|
39
|
+
|
40
|
+
div.day {
|
41
|
+
margin: 0;
|
42
|
+
height: inherit;
|
43
|
+
width: inherit;
|
44
|
+
display: block;
|
45
|
+
}
|
46
|
+
|
47
|
+
div.event {
|
48
|
+
margin: 2px, 10px;
|
49
|
+
border: 1px solid;
|
50
|
+
border-color: #d1d1d1;
|
51
|
+
padding: 2px 3px;
|
52
|
+
background: #ffffff;
|
53
|
+
}
|
54
|
+
|
55
|
+
.calendar-header {
|
56
|
+
button {
|
57
|
+
font-size: 20px;
|
58
|
+
width: 100%;
|
59
|
+
}
|
60
|
+
|
61
|
+
.calendar-title {
|
62
|
+
font-size: 24px;
|
63
|
+
text-align: center;
|
64
|
+
}
|
65
|
+
|
66
|
+
.previous {
|
67
|
+
padding-left: 0px;
|
68
|
+
padding-right: 20px;
|
69
|
+
}
|
70
|
+
|
71
|
+
.next {
|
72
|
+
padding-left: 20px;
|
73
|
+
padding-right: 0px;
|
74
|
+
}
|
75
|
+
}
|
76
|
+
}
|
@@ -0,0 +1 @@
|
|
1
|
+
export const CALENDAR_EVENT = 'CALENDAR_EVENT';
|
@@ -0,0 +1,77 @@
|
|
1
|
+
import { chunk, times } from 'lodash';
|
2
|
+
|
3
|
+
export const addDays = (date, days) => {
|
4
|
+
let result = new Date(date.valueOf());
|
5
|
+
result.setDate(result.getDate() + days);
|
6
|
+
return result;
|
7
|
+
};
|
8
|
+
|
9
|
+
export const addMonths = (date, months) => {
|
10
|
+
let result = new Date(date.valueOf());
|
11
|
+
result.setMonth(result.getMonth() + months);
|
12
|
+
return result;
|
13
|
+
};
|
14
|
+
|
15
|
+
export const startOfDay = (date = new Date()) => {
|
16
|
+
let result = new Date(date.valueOf());
|
17
|
+
result.setHours(0, 0, 0, 0);
|
18
|
+
return result;
|
19
|
+
};
|
20
|
+
|
21
|
+
export const endOfDay = (date = new Date()) => {
|
22
|
+
let result = new Date(date.valueOf());
|
23
|
+
result.setHours(23, 59, 59, 999);
|
24
|
+
return result;
|
25
|
+
};
|
26
|
+
|
27
|
+
export const isEqualDate = (a, b) => {
|
28
|
+
return a.getFullYear() === b.getFullYear() &&
|
29
|
+
a.getMonth() === b.getMonth() &&
|
30
|
+
a.getDate() === b.getDate();
|
31
|
+
};
|
32
|
+
|
33
|
+
export const startOfWeek = (date, weekStartsOn = 0) => {
|
34
|
+
let day = date.getDay();
|
35
|
+
let diff = day >= weekStartsOn ? day - weekStartsOn : 7 - weekStartsOn + day;
|
36
|
+
let result = new Date(date.valueOf());
|
37
|
+
result.setDate(result.getDate() - diff);
|
38
|
+
result.setHours(0, 0, 0, 0);
|
39
|
+
return result;
|
40
|
+
};
|
41
|
+
|
42
|
+
export const endOfWeek = (date, weekStartsOn = 0) => {
|
43
|
+
let day = date.getDay();
|
44
|
+
let diff = day >= weekStartsOn ? 6 - day + weekStartsOn : weekStartsOn - day - 1;
|
45
|
+
let result = new Date(date.valueOf());
|
46
|
+
result.setDate(result.getDate() + diff);
|
47
|
+
result.setHours(23, 59, 59, 999);
|
48
|
+
return result;
|
49
|
+
};
|
50
|
+
|
51
|
+
export const startOfMonth = (date) => {
|
52
|
+
return new Date(date.getFullYear(), date.getMonth(), 1);
|
53
|
+
};
|
54
|
+
|
55
|
+
export const endOfMonth = (date) => {
|
56
|
+
let result = new Date(date.getFullYear(), date.getMonth() + 1, 0);
|
57
|
+
result.setHours(23, 59, 59, 999);
|
58
|
+
return result;
|
59
|
+
};
|
60
|
+
|
61
|
+
export const getWeekArray = (weekStartsOn, locale) => {
|
62
|
+
let weekStart = startOfWeek(new Date(), weekStartsOn);
|
63
|
+
|
64
|
+
return times(7, i => Intl.DateTimeFormat(locale, { weekday: 'long' })
|
65
|
+
.format(addDays(weekStart, i)));
|
66
|
+
};
|
67
|
+
|
68
|
+
export const getWeeksOfMonth = (date, weekStartsOn = 0) => {
|
69
|
+
let start = startOfMonth(date);
|
70
|
+
let end = endOfMonth(date);
|
71
|
+
let offset = start.getDay() - weekStartsOn;
|
72
|
+
let days = offset + end.getDate() + (6 + weekStartsOn - end.getDay())
|
73
|
+
|
74
|
+
return chunk(times(days, i => addDays(start, i - offset)), 7);
|
75
|
+
};
|
76
|
+
|
77
|
+
|
@@ -0,0 +1,25 @@
|
|
1
|
+
import React from 'react';
|
2
|
+
import PropTypes from 'prop-types';
|
3
|
+
import { useDrag, useDrop } from 'react-dnd';
|
4
|
+
import { CALENDAR_EVENT } from './CalendarConstants';
|
5
|
+
|
6
|
+
const Event = (props) => {
|
7
|
+
|
8
|
+
const [, drag] = useDrag({
|
9
|
+
type: CALENDAR_EVENT,
|
10
|
+
item: {
|
11
|
+
type: CALENDAR_EVENT,
|
12
|
+
...props,
|
13
|
+
},
|
14
|
+
});
|
15
|
+
|
16
|
+
const { title } = props;
|
17
|
+
|
18
|
+
return (
|
19
|
+
<div ref={drag} className='event'>
|
20
|
+
{title}
|
21
|
+
</div>
|
22
|
+
);
|
23
|
+
};
|
24
|
+
|
25
|
+
export default Event;
|
@@ -0,0 +1,51 @@
|
|
1
|
+
import React, { useState } from 'react';
|
2
|
+
import { useDrop } from 'react-dnd';
|
3
|
+
import classNames from 'classnames';
|
4
|
+
import {
|
5
|
+
isEqualDate,
|
6
|
+
startOfDay,
|
7
|
+
endOfDay,
|
8
|
+
} from '../CalendarHelpers';
|
9
|
+
import Event from '../Event';
|
10
|
+
import { CALENDAR_EVENT } from '../CalendarConstants';
|
11
|
+
|
12
|
+
const Day = (props) => {
|
13
|
+
const { date, enabled, onEventMoved } = props;
|
14
|
+
|
15
|
+
const events = props.events.filter(event => isEqualDate(date, event.start));
|
16
|
+
|
17
|
+
const [{ canDrop, isOver }, drop] = useDrop({
|
18
|
+
accept: CALENDAR_EVENT,
|
19
|
+
drop: (event, monitor) => {
|
20
|
+
const start = new Date(event.start);
|
21
|
+
const end = new Date(event.end);
|
22
|
+
|
23
|
+
let delta = Math.ceil((date - start) / (1000 * 60 * 60 * 24));
|
24
|
+
|
25
|
+
start.setDate(start.getDate() + delta);
|
26
|
+
end.setDate(end.getDate() + delta);
|
27
|
+
|
28
|
+
onEventMoved({...event, start, end});
|
29
|
+
},
|
30
|
+
canDrop: (item, monitor) => (enabled),
|
31
|
+
});
|
32
|
+
|
33
|
+
const classes = {
|
34
|
+
disabled: !enabled,
|
35
|
+
enabled: enabled,
|
36
|
+
today: isEqualDate(date, new Date()),
|
37
|
+
past: date < startOfDay(),
|
38
|
+
future: date > endOfDay(),
|
39
|
+
}
|
40
|
+
|
41
|
+
return (
|
42
|
+
<td className={classNames(`day wday-${date.getDay()}`, classes)} data-day={date.getDate()}>
|
43
|
+
<h6>{date.getDate()}</h6>
|
44
|
+
<div ref={drop} className="day">
|
45
|
+
{events.map((event) => (<Event {...event} />))}
|
46
|
+
</div>
|
47
|
+
</td>
|
48
|
+
);
|
49
|
+
};
|
50
|
+
|
51
|
+
export default Day;
|
@@ -0,0 +1,77 @@
|
|
1
|
+
import React, { useState, useEffect } from 'react';
|
2
|
+
import { useDrop } from 'react-dnd';
|
3
|
+
import { Grid, Button, Icon } from 'patternfly-react';
|
4
|
+
|
5
|
+
import {
|
6
|
+
addMonths,
|
7
|
+
startOfMonth,
|
8
|
+
endOfMonth,
|
9
|
+
} from '../CalendarHelpers';
|
10
|
+
import { CALENDAR_EVENT } from '../CalendarConstants';
|
11
|
+
|
12
|
+
const Header = ({ start, end, date, setDate, setView, locale, weekStartsOn }) => {
|
13
|
+
const formatter = Intl.DateTimeFormat(locale, { month: 'long', year: 'numeric' })
|
14
|
+
|
15
|
+
const previousDate = addMonths(date, -1);
|
16
|
+
const previousDisabled = startOfMonth(date) < start;
|
17
|
+
const nextDate = addMonths(date, 1);
|
18
|
+
const nextDisabled = endOfMonth(date) > end;
|
19
|
+
|
20
|
+
const [wait, setWait] = useState(false);
|
21
|
+
|
22
|
+
useEffect(() => {
|
23
|
+
if (wait) {
|
24
|
+
setTimeout(() => {
|
25
|
+
setWait(false);
|
26
|
+
}, 1000);
|
27
|
+
}
|
28
|
+
}, [wait]);
|
29
|
+
|
30
|
+
const gotoPreviousMonth = () => {
|
31
|
+
setDate(previousDate);
|
32
|
+
};
|
33
|
+
|
34
|
+
const gotoNextMonth = () => {
|
35
|
+
setDate(nextDate);
|
36
|
+
};
|
37
|
+
|
38
|
+
const [, previous] = useDrop({
|
39
|
+
accept: CALENDAR_EVENT,
|
40
|
+
hover: (item, monitor) => {
|
41
|
+
if (!wait) {
|
42
|
+
gotoPreviousMonth();
|
43
|
+
setWait(true);
|
44
|
+
}
|
45
|
+
},
|
46
|
+
});
|
47
|
+
|
48
|
+
const [, next] = useDrop({
|
49
|
+
accept: CALENDAR_EVENT,
|
50
|
+
hover: (item, monitor) => {
|
51
|
+
if (!wait) {
|
52
|
+
gotoNextMonth();
|
53
|
+
setWait(true);
|
54
|
+
}
|
55
|
+
},
|
56
|
+
});
|
57
|
+
|
58
|
+
return (
|
59
|
+
<div className="calendar-header">
|
60
|
+
<span className="previous col-md-2" ref={previous}>
|
61
|
+
<Button variant="secondary" onClick={gotoPreviousMonth} disabled={previousDisabled} size="lg">
|
62
|
+
<Icon type="fa" name="angle-left" /> {formatter.format(previousDate)}
|
63
|
+
</Button>
|
64
|
+
</span>
|
65
|
+
<span className="calendar-title col-md-8">
|
66
|
+
{formatter.format(date)}
|
67
|
+
</span>
|
68
|
+
<span className="next col-md-2" ref={next}>
|
69
|
+
<Button variant="secondary" onClick={gotoNextMonth} disabled={nextDisabled} size="lg">
|
70
|
+
{formatter.format(nextDate)} <Icon type="fa" name="angle-right" />
|
71
|
+
</Button>
|
72
|
+
</span>
|
73
|
+
</div>
|
74
|
+
);
|
75
|
+
};
|
76
|
+
|
77
|
+
export default Header;
|
@@ -0,0 +1,47 @@
|
|
1
|
+
import React from 'react';
|
2
|
+
import { Grid } from 'patternfly-react';
|
3
|
+
|
4
|
+
import Day from './Day';
|
5
|
+
import Header from './Header';
|
6
|
+
import {
|
7
|
+
getWeeksOfMonth,
|
8
|
+
getWeekArray,
|
9
|
+
isEqualDate,
|
10
|
+
} from '../CalendarHelpers';
|
11
|
+
|
12
|
+
const Month = ({start, end, date, setDate, setView, events, onEventMoved, locale, weekStartsOn}) => {
|
13
|
+
const weeks = getWeeksOfMonth(date, weekStartsOn);
|
14
|
+
const daysOfTheWeek = getWeekArray(weekStartsOn, locale);
|
15
|
+
|
16
|
+
const isDayEnabled = (date) => {
|
17
|
+
return (start <= date && end > date);
|
18
|
+
};
|
19
|
+
|
20
|
+
return (
|
21
|
+
<div>
|
22
|
+
<Header start={start} end={end} date={date} setDate={setDate} setView={setView} locale={locale} weekStartsOn={weekStartsOn}/>
|
23
|
+
<table className="calendar-view table table-bordered table-fixed">
|
24
|
+
<thead>
|
25
|
+
<tr>
|
26
|
+
{daysOfTheWeek.map((day, index) => (
|
27
|
+
<th key={index} className="dow">
|
28
|
+
{day}
|
29
|
+
</th>
|
30
|
+
))}
|
31
|
+
</tr>
|
32
|
+
</thead>
|
33
|
+
<tbody>
|
34
|
+
{weeks.map(week => (
|
35
|
+
<tr className="week">
|
36
|
+
{week.map(day => (
|
37
|
+
<Day key={day} date={day} enabled={isDayEnabled(day)} events={events} onEventMoved={onEventMoved} />
|
38
|
+
))}
|
39
|
+
</tr>
|
40
|
+
))}
|
41
|
+
</tbody>
|
42
|
+
</table>
|
43
|
+
</div>
|
44
|
+
);
|
45
|
+
};
|
46
|
+
|
47
|
+
export default Month;
|
@@ -0,0 +1 @@
|
|
1
|
+
export { default } from './Month';
|
@@ -0,0 +1,20 @@
|
|
1
|
+
import Month from './Month';
|
2
|
+
import Week from './Week';
|
3
|
+
import Day from './Day';
|
4
|
+
|
5
|
+
export const views = {
|
6
|
+
MONTH: 'month',
|
7
|
+
WEEK: 'week',
|
8
|
+
DAY: 'day',
|
9
|
+
};
|
10
|
+
|
11
|
+
export const VIEWS = {
|
12
|
+
[views.MONTH]: Month,
|
13
|
+
[views.WEEK]: Week,
|
14
|
+
[views.DAY]: Day,
|
15
|
+
};
|
16
|
+
|
17
|
+
export const getView = (view) => {
|
18
|
+
return VIEWS[view];
|
19
|
+
};
|
20
|
+
|
@@ -0,0 +1,26 @@
|
|
1
|
+
import React, { useState } from 'react';
|
2
|
+
import PropTypes from 'prop-types';
|
3
|
+
import { classnames } from 'classname';
|
4
|
+
|
5
|
+
const OutputLine = ({index, type, children}) => {
|
6
|
+
return (
|
7
|
+
<div className={classnames('line', type)} >
|
8
|
+
<span className="counter">{index}</span>
|
9
|
+
<div className="content">
|
10
|
+
{children}
|
11
|
+
</div>
|
12
|
+
</div>
|
13
|
+
);
|
14
|
+
};
|
15
|
+
|
16
|
+
OutputLine.propTypes = {
|
17
|
+
index: PropTypes.number.isRequired,
|
18
|
+
type: PropTypes.string,
|
19
|
+
timestamp: PropTypes.number,
|
20
|
+
};
|
21
|
+
|
22
|
+
OutputLine.defaultProps = {
|
23
|
+
type: 'stdout',
|
24
|
+
};
|
25
|
+
|
26
|
+
export default OutputLine;
|