upright 0.1.0

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.
Files changed (121) hide show
  1. checksums.yaml +7 -0
  2. data/LICENSE.md +10 -0
  3. data/README.md +455 -0
  4. data/Rakefile +6 -0
  5. data/app/assets/stylesheets/upright/_global.css +104 -0
  6. data/app/assets/stylesheets/upright/artifact.css +148 -0
  7. data/app/assets/stylesheets/upright/base.css +68 -0
  8. data/app/assets/stylesheets/upright/buttons.css +21 -0
  9. data/app/assets/stylesheets/upright/dashboard.css +287 -0
  10. data/app/assets/stylesheets/upright/forms.css +104 -0
  11. data/app/assets/stylesheets/upright/header.css +124 -0
  12. data/app/assets/stylesheets/upright/layout.css +100 -0
  13. data/app/assets/stylesheets/upright/map.css +25 -0
  14. data/app/assets/stylesheets/upright/pagination.css +45 -0
  15. data/app/assets/stylesheets/upright/probes.css +72 -0
  16. data/app/assets/stylesheets/upright/reset.css +26 -0
  17. data/app/assets/stylesheets/upright/tables.css +63 -0
  18. data/app/assets/stylesheets/upright/typography.css +27 -0
  19. data/app/assets/stylesheets/upright/uptime-bars.css +154 -0
  20. data/app/controllers/concerns/upright/authentication.rb +21 -0
  21. data/app/controllers/concerns/upright/subdomain_scoping.rb +18 -0
  22. data/app/controllers/upright/alertmanager_proxy_controller.rb +21 -0
  23. data/app/controllers/upright/application_controller.rb +12 -0
  24. data/app/controllers/upright/artifacts_controller.rb +5 -0
  25. data/app/controllers/upright/dashboards/uptimes_controller.rb +6 -0
  26. data/app/controllers/upright/jobs_controller.rb +4 -0
  27. data/app/controllers/upright/probe_results_controller.rb +17 -0
  28. data/app/controllers/upright/prometheus_proxy_controller.rb +62 -0
  29. data/app/controllers/upright/sessions_controller.rb +29 -0
  30. data/app/controllers/upright/sites_controller.rb +5 -0
  31. data/app/helpers/upright/application_helper.rb +11 -0
  32. data/app/helpers/upright/dashboards_helper.rb +31 -0
  33. data/app/helpers/upright/probe_results_helper.rb +49 -0
  34. data/app/javascript/upright/application.js +2 -0
  35. data/app/javascript/upright/controllers/application.js +5 -0
  36. data/app/javascript/upright/controllers/form_controller.js +7 -0
  37. data/app/javascript/upright/controllers/index.js +4 -0
  38. data/app/javascript/upright/controllers/popover_controller.js +15 -0
  39. data/app/javascript/upright/controllers/probe_results_chart_controller.js +79 -0
  40. data/app/javascript/upright/controllers/results_table_controller.js +16 -0
  41. data/app/javascript/upright/controllers/sites_map_controller.js +33 -0
  42. data/app/jobs/upright/application_job.rb +2 -0
  43. data/app/jobs/upright/probe_check_job.rb +42 -0
  44. data/app/models/concerns/upright/exception_recording.rb +38 -0
  45. data/app/models/concerns/upright/playwright/form_authentication.rb +27 -0
  46. data/app/models/concerns/upright/playwright/helpers.rb +7 -0
  47. data/app/models/concerns/upright/playwright/lifecycle.rb +44 -0
  48. data/app/models/concerns/upright/playwright/logging.rb +87 -0
  49. data/app/models/concerns/upright/playwright/otel_tracing.rb +137 -0
  50. data/app/models/concerns/upright/playwright/video_recording.rb +60 -0
  51. data/app/models/concerns/upright/probe_yaml_source.rb +10 -0
  52. data/app/models/concerns/upright/probeable.rb +125 -0
  53. data/app/models/concerns/upright/staggerable.rb +22 -0
  54. data/app/models/concerns/upright/traceroute/otel_tracing.rb +108 -0
  55. data/app/models/upright/application_record.rb +3 -0
  56. data/app/models/upright/artifact.rb +61 -0
  57. data/app/models/upright/current.rb +9 -0
  58. data/app/models/upright/http/request.rb +59 -0
  59. data/app/models/upright/http/response.rb +55 -0
  60. data/app/models/upright/playwright/authenticator/base.rb +128 -0
  61. data/app/models/upright/playwright/storage_state.rb +31 -0
  62. data/app/models/upright/probe_result.rb +31 -0
  63. data/app/models/upright/probes/http_probe.rb +102 -0
  64. data/app/models/upright/probes/playwright/base.rb +48 -0
  65. data/app/models/upright/probes/smtp_probe.rb +48 -0
  66. data/app/models/upright/probes/traceroute_probe.rb +32 -0
  67. data/app/models/upright/probes/uptime/summary.rb +36 -0
  68. data/app/models/upright/probes/uptime.rb +36 -0
  69. data/app/models/upright/traceroute/hop.rb +49 -0
  70. data/app/models/upright/traceroute/ip_metadata_lookup.rb +107 -0
  71. data/app/models/upright/traceroute/mtr_parser.rb +47 -0
  72. data/app/models/upright/traceroute/result.rb +57 -0
  73. data/app/models/upright/user.rb +14 -0
  74. data/app/views/layouts/upright/_header.html.erb +23 -0
  75. data/app/views/layouts/upright/application.html.erb +25 -0
  76. data/app/views/upright/active_storage/attachments/_attachment.html.erb +21 -0
  77. data/app/views/upright/alertmanager_proxy/show.html.erb +1 -0
  78. data/app/views/upright/artifacts/show.html.erb +9 -0
  79. data/app/views/upright/dashboards/_uptime_bars.html.erb +17 -0
  80. data/app/views/upright/dashboards/_uptime_probe_row.html.erb +22 -0
  81. data/app/views/upright/dashboards/uptimes/show.html.erb +17 -0
  82. data/app/views/upright/jobs/show.html.erb +1 -0
  83. data/app/views/upright/probe_results/_pagination.html.erb +19 -0
  84. data/app/views/upright/probe_results/index.html.erb +72 -0
  85. data/app/views/upright/prometheus_proxy/show.html.erb +1 -0
  86. data/app/views/upright/sessions/new.html.erb +6 -0
  87. data/app/views/upright/sites/index.html.erb +22 -0
  88. data/config/brakeman.ignore +39 -0
  89. data/config/ci.rb +7 -0
  90. data/config/importmap.rb +18 -0
  91. data/config/routes.rb +41 -0
  92. data/db/migrate/20250114000001_create_upright_probe_results.rb +19 -0
  93. data/lib/generators/upright/install/install_generator.rb +83 -0
  94. data/lib/generators/upright/install/templates/alertmanager.yml +14 -0
  95. data/lib/generators/upright/install/templates/deploy.yml +118 -0
  96. data/lib/generators/upright/install/templates/development_alertmanager.yml +11 -0
  97. data/lib/generators/upright/install/templates/development_prometheus.yml +12 -0
  98. data/lib/generators/upright/install/templates/docker-compose.yml +38 -0
  99. data/lib/generators/upright/install/templates/http_probes.yml +14 -0
  100. data/lib/generators/upright/install/templates/omniauth.rb +8 -0
  101. data/lib/generators/upright/install/templates/otel_collector.yml +24 -0
  102. data/lib/generators/upright/install/templates/prometheus.yml +10 -0
  103. data/lib/generators/upright/install/templates/puma.rb +40 -0
  104. data/lib/generators/upright/install/templates/sites.yml +26 -0
  105. data/lib/generators/upright/install/templates/smtp_probes.yml +9 -0
  106. data/lib/generators/upright/install/templates/upright.rb +21 -0
  107. data/lib/generators/upright/install/templates/upright.rules.yml +256 -0
  108. data/lib/generators/upright/playwright_probe/playwright_probe_generator.rb +30 -0
  109. data/lib/generators/upright/playwright_probe/templates/authenticator.rb.tt +14 -0
  110. data/lib/generators/upright/playwright_probe/templates/probe.rb.tt +14 -0
  111. data/lib/omniauth/strategies/static_credentials.rb +57 -0
  112. data/lib/tasks/upright_tasks.rake +4 -0
  113. data/lib/upright/configuration.rb +106 -0
  114. data/lib/upright/engine.rb +157 -0
  115. data/lib/upright/metrics.rb +62 -0
  116. data/lib/upright/playwright/collect_performance_metrics.js +36 -0
  117. data/lib/upright/site.rb +49 -0
  118. data/lib/upright/tracing.rb +49 -0
  119. data/lib/upright/version.rb +3 -0
  120. data/lib/upright.rb +68 -0
  121. metadata +513 -0
@@ -0,0 +1,124 @@
1
+ @layer components {
2
+ .header {
3
+ align-items: center;
4
+ background-color: var(--color-ink-lightest);
5
+ border-bottom: var(--border);
6
+ display: flex;
7
+ gap: calc(var(--inline-space) * 2);
8
+ justify-content: space-between;
9
+ padding: var(--block-space) calc(var(--inline-space) * 2);
10
+ }
11
+
12
+ .header__left {
13
+ align-items: center;
14
+ display: flex;
15
+ gap: var(--inline-space);
16
+ }
17
+
18
+ .header__logo {
19
+ color: var(--color-ink);
20
+ font-size: var(--text-normal);
21
+ font-weight: 600;
22
+ letter-spacing: 0.05em;
23
+ text-decoration: none;
24
+ text-transform: uppercase;
25
+
26
+ &:hover {
27
+ color: var(--color-link);
28
+ text-decoration: none;
29
+ }
30
+ }
31
+
32
+ .header__site {
33
+ color: var(--color-positive);
34
+ font-size: var(--text-x-small);
35
+ font-weight: 500;
36
+ letter-spacing: 0.06em;
37
+ margin-right: calc(var(--inline-space) * 2);
38
+ text-transform: uppercase;
39
+ }
40
+
41
+ .header__nav {
42
+ align-items: center;
43
+ display: flex;
44
+ flex: 1;
45
+ gap: calc(var(--inline-space) * 3);
46
+ }
47
+
48
+ .header__link {
49
+ align-items: center;
50
+ color: var(--color-ink-dark);
51
+ display: flex;
52
+ font-size: var(--text-x-small);
53
+ font-weight: 500;
54
+ gap: 0.25em;
55
+ letter-spacing: 0.06em;
56
+ text-transform: uppercase;
57
+
58
+ &:hover {
59
+ color: var(--color-ink);
60
+ }
61
+
62
+ &.active {
63
+ color: var(--color-positive);
64
+ }
65
+ }
66
+
67
+ .header__divider {
68
+ background: var(--color-ink-light);
69
+ height: 1em;
70
+ width: 1px;
71
+ }
72
+
73
+ .header__user {
74
+ align-items: center;
75
+ display: flex;
76
+ gap: var(--inline-space);
77
+ }
78
+
79
+ .main {
80
+ padding: calc(var(--block-space) * 2);
81
+ }
82
+
83
+ @media (max-width: 85ch) {
84
+ .header {
85
+ flex-direction: column;
86
+ align-items: stretch;
87
+ gap: var(--block-space);
88
+ padding: var(--block-space);
89
+ }
90
+
91
+ .header__left {
92
+ justify-content: space-between;
93
+ }
94
+
95
+ .header__nav {
96
+ flex-wrap: wrap;
97
+ gap: calc(var(--inline-space) * 2);
98
+ }
99
+
100
+ .header__user {
101
+ justify-content: space-between;
102
+ padding-top: var(--block-space);
103
+ border-top: var(--border);
104
+ }
105
+ }
106
+
107
+ @media (max-width: 55ch) {
108
+ .header {
109
+ padding: calc(var(--block-space) * 0.75);
110
+ }
111
+
112
+ .header__nav {
113
+ gap: var(--inline-space);
114
+ }
115
+
116
+ .header__link {
117
+ font-size: calc(var(--text-x-small) * 0.9);
118
+ }
119
+
120
+ .main {
121
+ padding: var(--block-space);
122
+ }
123
+ }
124
+ }
@@ -0,0 +1,100 @@
1
+ @layer components {
2
+ .service-frame {
3
+ border: none;
4
+ display: block;
5
+ height: calc(100dvh - 3.5rem);
6
+ width: 100%;
7
+ }
8
+
9
+ .main:has(.service-frame) {
10
+ overflow: hidden;
11
+ padding: 0;
12
+ }
13
+
14
+ body:has(.service-frame) {
15
+ overflow: hidden;
16
+ }
17
+
18
+ .container {
19
+ display: flex;
20
+ gap: calc(var(--inline-space) * 2);
21
+ margin: 0 auto;
22
+ max-width: var(--main-width);
23
+ }
24
+
25
+ aside {
26
+ background: oklch(var(--lch-ink-lightest) / 85%);
27
+ border-radius: 0.25rem;
28
+ box-shadow: var(--shadow);
29
+ height: fit-content;
30
+ min-width: 180px;
31
+ padding: var(--block-space);
32
+
33
+ ul {
34
+ list-style: none;
35
+ margin: 0;
36
+ padding: 0;
37
+ }
38
+
39
+ li {
40
+ margin-bottom: 2px;
41
+ }
42
+
43
+ a {
44
+ border-radius: 0.125rem;
45
+ display: block;
46
+ font-size: var(--text-small);
47
+ padding: calc(var(--block-space) / 2);
48
+ transition: background-color 100ms;
49
+
50
+ &:hover {
51
+ background-color: var(--color-ink-lighter);
52
+ text-decoration: none;
53
+ }
54
+
55
+ &.active {
56
+ background-color: var(--color-ink);
57
+ color: var(--color-ink-inverted);
58
+ font-weight: 500;
59
+
60
+ &:hover {
61
+ background-color: var(--color-link);
62
+ }
63
+ }
64
+ }
65
+ }
66
+
67
+ main {
68
+ flex: 1;
69
+ }
70
+
71
+ @media (max-width: 140ch) {
72
+ .container {
73
+ flex-direction: column;
74
+ }
75
+
76
+ aside {
77
+ min-width: auto;
78
+ width: 100%;
79
+ }
80
+
81
+ aside ul {
82
+ display: flex;
83
+ flex-wrap: nowrap;
84
+ gap: 0.25em;
85
+ }
86
+
87
+ aside li {
88
+ margin-bottom: 0;
89
+ }
90
+
91
+ aside a {
92
+ padding: 0.4em 0.6em;
93
+ font-size: var(--text-x-small);
94
+ }
95
+
96
+ .main {
97
+ padding: var(--block-space);
98
+ }
99
+ }
100
+ }
@@ -0,0 +1,25 @@
1
+ @layer components {
2
+ #sites-map {
3
+ border-radius: 0.25rem;
4
+ box-shadow: var(--shadow);
5
+ height: 400px;
6
+ margin-bottom: calc(var(--block-space) * 2);
7
+ width: 100%;
8
+
9
+ .leaflet-tile-pane img {
10
+ max-width: none !important;
11
+ }
12
+ }
13
+
14
+ @media (max-width: 140ch) {
15
+ #sites-map {
16
+ height: 300px;
17
+ }
18
+ }
19
+
20
+ @media (max-width: 55ch) {
21
+ #sites-map {
22
+ height: 250px;
23
+ }
24
+ }
25
+ }
@@ -0,0 +1,45 @@
1
+ @layer components {
2
+ .pagination {
3
+ align-items: center;
4
+ display: flex;
5
+ gap: calc(var(--inline-space) * 2);
6
+ justify-content: center;
7
+ margin-top: calc(var(--block-space) * 2);
8
+ }
9
+
10
+ .pagination__prev,
11
+ .pagination__next {
12
+ background: var(--color-ink-lightest);
13
+ border-radius: 0.125rem;
14
+ border: var(--border);
15
+ color: var(--color-ink);
16
+ font-size: var(--text-x-small);
17
+ font-weight: 500;
18
+ letter-spacing: 0.02em;
19
+ padding: 0.5em 1em;
20
+ text-transform: uppercase;
21
+
22
+ &:hover:not(.pagination__prev--disabled):not(.pagination__next--disabled) {
23
+ background: var(--color-ink-lighter);
24
+ text-decoration: none;
25
+ }
26
+ }
27
+
28
+ .pagination__prev--disabled,
29
+ .pagination__next--disabled {
30
+ color: var(--color-ink-medium);
31
+ cursor: not-allowed;
32
+ }
33
+
34
+ .pagination__info {
35
+ color: var(--color-ink-dark);
36
+ font-size: var(--text-small);
37
+ }
38
+
39
+ @media (max-width: 140ch) {
40
+ .pagination {
41
+ flex-wrap: wrap;
42
+ gap: var(--block-space);
43
+ }
44
+ }
45
+ }
@@ -0,0 +1,72 @@
1
+ @layer components {
2
+ .probe__cell {
3
+ display: flex;
4
+ gap: 0.5em;
5
+ }
6
+
7
+ .probe__icon {
8
+ flex-shrink: 0;
9
+ }
10
+
11
+ .probe__name {
12
+ font-weight: 500;
13
+ }
14
+
15
+ .probe__target {
16
+ color: var(--color-ink-medium);
17
+ font-size: var(--text-x-small);
18
+ margin-top: 0.125em;
19
+ }
20
+
21
+ .probe__badge {
22
+ background: var(--color-ink-lighter);
23
+ border-radius: 0.125rem;
24
+ color: var(--color-ink-dark);
25
+ font-size: var(--text-x-small);
26
+ padding: 0.25em 0.5em;
27
+ text-transform: uppercase;
28
+ }
29
+
30
+ .status--ok {
31
+ color: var(--color-positive);
32
+ font-weight: 500;
33
+ }
34
+
35
+ .status--fail {
36
+ color: var(--color-negative);
37
+ font-weight: 600;
38
+ }
39
+
40
+ .probe-results__chart {
41
+ background: oklch(var(--lch-ink-lightest) / 85%);
42
+ border-radius: 0.25rem;
43
+ box-shadow: var(--shadow);
44
+ margin-bottom: calc(var(--block-space) * 1.5);
45
+ padding: var(--block-space);
46
+
47
+ circle {
48
+ stroke: none;
49
+ }
50
+
51
+ text {
52
+ font-family: var(--font-mono);
53
+ font-size: var(--text-x-small);
54
+ }
55
+ }
56
+
57
+ @media (max-width: 140ch) {
58
+ .probe__target {
59
+ display: none;
60
+ }
61
+
62
+ .probe__cell {
63
+ gap: 0.25em;
64
+ }
65
+ }
66
+
67
+ @media (max-width: 55ch) {
68
+ .probe__name {
69
+ word-break: break-word;
70
+ }
71
+ }
72
+ }
@@ -0,0 +1,26 @@
1
+ @layer reset {
2
+ *,
3
+ *::before,
4
+ *::after {
5
+ box-sizing: border-box;
6
+ }
7
+
8
+ body,
9
+ h1,
10
+ h2,
11
+ h3,
12
+ h4,
13
+ p {
14
+ margin: 0;
15
+ }
16
+
17
+ summary {
18
+ &::-webkit-details-marker {
19
+ display: none;
20
+ }
21
+
22
+ &::marker {
23
+ content: "";
24
+ }
25
+ }
26
+ }
@@ -0,0 +1,63 @@
1
+ @layer components {
2
+ table {
3
+ background: oklch(var(--lch-ink-lightest) / 85%);
4
+ border-collapse: collapse;
5
+ border-radius: 0.25rem;
6
+ box-shadow: var(--shadow);
7
+ overflow: hidden;
8
+ width: 100%;
9
+ }
10
+
11
+ thead {
12
+ background: var(--color-ink-lighter);
13
+ }
14
+
15
+ th {
16
+ font-size: var(--text-x-small);
17
+ font-weight: 500;
18
+ letter-spacing: 0.08em;
19
+ padding: calc(var(--block-space) * 0.75) var(--block-space);
20
+ text-align: left;
21
+ text-transform: uppercase;
22
+ }
23
+
24
+ td {
25
+ border-top: var(--border);
26
+ font-size: var(--text-small);
27
+ padding: calc(var(--block-space) * 0.75) var(--block-space);
28
+ }
29
+
30
+ tbody tr {
31
+ cursor: pointer;
32
+ transition: background-color 100ms;
33
+
34
+ &:hover {
35
+ background-color: var(--color-canvas);
36
+ }
37
+ }
38
+
39
+ @media (max-width: 140ch) {
40
+ th, td {
41
+ padding: calc(var(--block-space) * 0.5) calc(var(--block-space) * 0.5);
42
+ font-size: var(--text-x-small);
43
+ }
44
+
45
+ /* Hide Duration column (3rd column) */
46
+ th:nth-child(3),
47
+ td:nth-child(3) {
48
+ display: none;
49
+ }
50
+ }
51
+
52
+ @media (max-width: 55ch) {
53
+ /* Hide Time column (4th column) */
54
+ th:nth-child(4),
55
+ td:nth-child(4) {
56
+ display: none;
57
+ }
58
+
59
+ th, td {
60
+ padding: calc(var(--block-space) * 0.4) calc(var(--block-space) * 0.4);
61
+ }
62
+ }
63
+ }
@@ -0,0 +1,27 @@
1
+ @layer base {
2
+ h1 {
3
+ font-size: var(--text-x-large);
4
+ font-weight: 500;
5
+ letter-spacing: -0.02em;
6
+ margin-bottom: var(--block-space);
7
+ text-transform: uppercase;
8
+ }
9
+
10
+ h2 {
11
+ font-size: var(--text-large);
12
+ font-weight: 500;
13
+ letter-spacing: -0.01em;
14
+ margin-bottom: var(--block-space);
15
+ }
16
+
17
+ p {
18
+ color: var(--color-ink-dark);
19
+ margin-bottom: calc(var(--block-space) * 1.5);
20
+ }
21
+
22
+ @media (max-width: 55ch) {
23
+ h1 {
24
+ font-size: var(--text-large);
25
+ }
26
+ }
27
+ }
@@ -0,0 +1,154 @@
1
+ @layer components {
2
+ .uptime-bars {
3
+ display: flex;
4
+ flex-direction: column;
5
+ }
6
+
7
+ .uptime-bars__header {
8
+ background: var(--color-ink-lighter);
9
+ display: grid;
10
+ font-size: var(--text-x-small);
11
+ font-weight: 500;
12
+ gap: var(--block-space);
13
+ grid-template-columns: minmax(200px, 1fr) 1fr auto;
14
+ letter-spacing: 0.04em;
15
+ padding: calc(var(--block-space) * 0.75) var(--block-space);
16
+ text-transform: uppercase;
17
+ }
18
+
19
+
20
+ .uptime-bars__row {
21
+ border-top: 1px solid var(--color-ink-lighter);
22
+ display: grid;
23
+ gap: var(--block-space);
24
+ grid-template-columns: minmax(200px, 1fr) 1fr auto;
25
+ padding: calc(var(--block-space) * 0.75) var(--block-space);
26
+ transition: background-color 100ms;
27
+ }
28
+
29
+ .uptime-bars__row:hover {
30
+ background: var(--color-canvas);
31
+ }
32
+
33
+ .uptime-bars__probe {
34
+ align-items: center;
35
+ color: inherit;
36
+ display: flex;
37
+ gap: 0.75em;
38
+ min-width: 0;
39
+
40
+ &:hover {
41
+ color: var(--color-link);
42
+ text-decoration: none;
43
+ }
44
+ }
45
+
46
+ .uptime-bars__probe .probe__badge {
47
+ flex-shrink: 0;
48
+ }
49
+
50
+ .uptime-bars__probe .probe__name {
51
+ overflow: hidden;
52
+ text-overflow: ellipsis;
53
+ white-space: nowrap;
54
+ }
55
+
56
+ .uptime-bars__days {
57
+ align-items: center;
58
+ display: flex;
59
+ gap: 2px;
60
+ }
61
+
62
+ .uptime-bars__uptime {
63
+ font-size: var(--text-small);
64
+ font-weight: 500;
65
+ text-align: right;
66
+ width: 7ch;
67
+ }
68
+
69
+ .uptime-bars__uptime--excellent {
70
+ color: var(--color-positive);
71
+ }
72
+
73
+ .uptime-bars__uptime--good {
74
+ color: oklch(70% 0.12 145);
75
+ }
76
+
77
+ .uptime-bars__uptime--warning {
78
+ color: oklch(75% 0.15 85);
79
+ }
80
+
81
+ .uptime-bars__uptime--critical,
82
+ .uptime-bars__uptime--down {
83
+ color: var(--color-negative);
84
+ }
85
+
86
+ /* Individual uptime bar (separate block) */
87
+ .uptime-bar {
88
+ border-radius: 2px;
89
+ cursor: default;
90
+ flex: 1;
91
+ height: 24px;
92
+ max-width: 12px;
93
+ min-width: 4px;
94
+ transition: transform 100ms, filter 100ms;
95
+ }
96
+
97
+ .uptime-bar:hover {
98
+ filter: brightness(1.1);
99
+ transform: scaleY(1.15);
100
+ }
101
+
102
+ .uptime-bar--excellent {
103
+ background: var(--color-positive);
104
+ }
105
+
106
+ .uptime-bar--good {
107
+ background: oklch(70% 0.14 145);
108
+ }
109
+
110
+ .uptime-bar--warning {
111
+ background: oklch(75% 0.15 85);
112
+ }
113
+
114
+ .uptime-bar--critical {
115
+ background: oklch(65% 0.16 42);
116
+ }
117
+
118
+ .uptime-bar--down {
119
+ background: var(--color-negative);
120
+ }
121
+
122
+ .uptime-bar--no-data {
123
+ background: var(--color-ink-light);
124
+ opacity: 0.4;
125
+ }
126
+
127
+ @media (max-width: 85ch) {
128
+ .uptime-bars__header,
129
+ .uptime-bars__row {
130
+ grid-template-columns: minmax(120px, 150px) 1fr auto;
131
+ gap: calc(var(--block-space) * 0.5);
132
+ }
133
+
134
+ .uptime-bars__probe .probe__badge {
135
+ display: none;
136
+ }
137
+ }
138
+
139
+ @media (max-width: 55ch) {
140
+ .uptime-bars__header,
141
+ .uptime-bars__row {
142
+ grid-template-columns: minmax(80px, 120px) 1fr auto;
143
+ padding: calc(var(--block-space) * 0.5);
144
+ }
145
+
146
+ .uptime-bar {
147
+ height: 18px;
148
+ }
149
+
150
+ .uptime-bars__days .uptime-bar:not(:nth-last-child(-n+7)) {
151
+ display: none;
152
+ }
153
+ }
154
+ }
@@ -0,0 +1,21 @@
1
+ module Upright::Authentication
2
+ extend ActiveSupport::Concern
3
+
4
+ included do
5
+ before_action :authenticate_user
6
+ helper_method :signed_in?
7
+ end
8
+
9
+ private
10
+ def authenticate_user
11
+ if session[:user_info].present?
12
+ Upright::Current.user = Upright::User.new(session[:user_info])
13
+ else
14
+ redirect_to new_admin_session_url(default_url_options.merge(subdomain: Upright.configuration.global_subdomain)), allow_other_host: true
15
+ end
16
+ end
17
+
18
+ def signed_in?
19
+ Upright::Current.user.present?
20
+ end
21
+ end
@@ -0,0 +1,18 @@
1
+ module Upright::SubdomainScoping
2
+ extend ActiveSupport::Concern
3
+
4
+ included do
5
+ before_action :redirect_to_app_subdomain, if: -> { request.subdomain.blank? }
6
+ before_action :set_current_subdomain
7
+ end
8
+
9
+ private
10
+ def redirect_to_app_subdomain
11
+ redirect_to root_url(default_url_options.merge(subdomain: Upright.configuration.global_subdomain)), allow_other_host: true
12
+ end
13
+
14
+ def set_current_subdomain
15
+ Upright::Current.subdomain = request.subdomain.presence
16
+ Upright::Current.site = Upright.find_site(Upright::Current.subdomain)
17
+ end
18
+ end
@@ -0,0 +1,21 @@
1
+ class Upright::AlertmanagerProxyController < Upright::ApplicationController
2
+ skip_forgery_protection
3
+
4
+ def show
5
+ end
6
+
7
+ def proxy
8
+ proxy_to_alertmanager request.fullpath.delete_prefix("/alertmanager")
9
+ end
10
+
11
+ private
12
+ def proxy_to_alertmanager(path, method: request.method, body: nil)
13
+ response = Faraday.new(url: alertmanager_url).run_request(method.downcase.to_sym, path, body, { "Content-Type" => request.content_type })
14
+
15
+ render body: response.body, status: response.status, content_type: response.headers["content-type"]
16
+ end
17
+
18
+ def alertmanager_url
19
+ ENV.fetch("ALERTMANAGER_URL", "http://localhost:9093")
20
+ end
21
+ end
@@ -0,0 +1,12 @@
1
+ class Upright::ApplicationController < ActionController::Base
2
+ include Upright::SubdomainScoping
3
+ include Upright::Authentication
4
+
5
+ helper :all
6
+ protect_from_forgery with: :exception
7
+
8
+ private
9
+ def default_url_options
10
+ Rails.application.routes.default_url_options
11
+ end
12
+ end