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.
- checksums.yaml +7 -0
- data/LICENSE.md +10 -0
- data/README.md +455 -0
- data/Rakefile +6 -0
- data/app/assets/stylesheets/upright/_global.css +104 -0
- data/app/assets/stylesheets/upright/artifact.css +148 -0
- data/app/assets/stylesheets/upright/base.css +68 -0
- data/app/assets/stylesheets/upright/buttons.css +21 -0
- data/app/assets/stylesheets/upright/dashboard.css +287 -0
- data/app/assets/stylesheets/upright/forms.css +104 -0
- data/app/assets/stylesheets/upright/header.css +124 -0
- data/app/assets/stylesheets/upright/layout.css +100 -0
- data/app/assets/stylesheets/upright/map.css +25 -0
- data/app/assets/stylesheets/upright/pagination.css +45 -0
- data/app/assets/stylesheets/upright/probes.css +72 -0
- data/app/assets/stylesheets/upright/reset.css +26 -0
- data/app/assets/stylesheets/upright/tables.css +63 -0
- data/app/assets/stylesheets/upright/typography.css +27 -0
- data/app/assets/stylesheets/upright/uptime-bars.css +154 -0
- data/app/controllers/concerns/upright/authentication.rb +21 -0
- data/app/controllers/concerns/upright/subdomain_scoping.rb +18 -0
- data/app/controllers/upright/alertmanager_proxy_controller.rb +21 -0
- data/app/controllers/upright/application_controller.rb +12 -0
- data/app/controllers/upright/artifacts_controller.rb +5 -0
- data/app/controllers/upright/dashboards/uptimes_controller.rb +6 -0
- data/app/controllers/upright/jobs_controller.rb +4 -0
- data/app/controllers/upright/probe_results_controller.rb +17 -0
- data/app/controllers/upright/prometheus_proxy_controller.rb +62 -0
- data/app/controllers/upright/sessions_controller.rb +29 -0
- data/app/controllers/upright/sites_controller.rb +5 -0
- data/app/helpers/upright/application_helper.rb +11 -0
- data/app/helpers/upright/dashboards_helper.rb +31 -0
- data/app/helpers/upright/probe_results_helper.rb +49 -0
- data/app/javascript/upright/application.js +2 -0
- data/app/javascript/upright/controllers/application.js +5 -0
- data/app/javascript/upright/controllers/form_controller.js +7 -0
- data/app/javascript/upright/controllers/index.js +4 -0
- data/app/javascript/upright/controllers/popover_controller.js +15 -0
- data/app/javascript/upright/controllers/probe_results_chart_controller.js +79 -0
- data/app/javascript/upright/controllers/results_table_controller.js +16 -0
- data/app/javascript/upright/controllers/sites_map_controller.js +33 -0
- data/app/jobs/upright/application_job.rb +2 -0
- data/app/jobs/upright/probe_check_job.rb +42 -0
- data/app/models/concerns/upright/exception_recording.rb +38 -0
- data/app/models/concerns/upright/playwright/form_authentication.rb +27 -0
- data/app/models/concerns/upright/playwright/helpers.rb +7 -0
- data/app/models/concerns/upright/playwright/lifecycle.rb +44 -0
- data/app/models/concerns/upright/playwright/logging.rb +87 -0
- data/app/models/concerns/upright/playwright/otel_tracing.rb +137 -0
- data/app/models/concerns/upright/playwright/video_recording.rb +60 -0
- data/app/models/concerns/upright/probe_yaml_source.rb +10 -0
- data/app/models/concerns/upright/probeable.rb +125 -0
- data/app/models/concerns/upright/staggerable.rb +22 -0
- data/app/models/concerns/upright/traceroute/otel_tracing.rb +108 -0
- data/app/models/upright/application_record.rb +3 -0
- data/app/models/upright/artifact.rb +61 -0
- data/app/models/upright/current.rb +9 -0
- data/app/models/upright/http/request.rb +59 -0
- data/app/models/upright/http/response.rb +55 -0
- data/app/models/upright/playwright/authenticator/base.rb +128 -0
- data/app/models/upright/playwright/storage_state.rb +31 -0
- data/app/models/upright/probe_result.rb +31 -0
- data/app/models/upright/probes/http_probe.rb +102 -0
- data/app/models/upright/probes/playwright/base.rb +48 -0
- data/app/models/upright/probes/smtp_probe.rb +48 -0
- data/app/models/upright/probes/traceroute_probe.rb +32 -0
- data/app/models/upright/probes/uptime/summary.rb +36 -0
- data/app/models/upright/probes/uptime.rb +36 -0
- data/app/models/upright/traceroute/hop.rb +49 -0
- data/app/models/upright/traceroute/ip_metadata_lookup.rb +107 -0
- data/app/models/upright/traceroute/mtr_parser.rb +47 -0
- data/app/models/upright/traceroute/result.rb +57 -0
- data/app/models/upright/user.rb +14 -0
- data/app/views/layouts/upright/_header.html.erb +23 -0
- data/app/views/layouts/upright/application.html.erb +25 -0
- data/app/views/upright/active_storage/attachments/_attachment.html.erb +21 -0
- data/app/views/upright/alertmanager_proxy/show.html.erb +1 -0
- data/app/views/upright/artifacts/show.html.erb +9 -0
- data/app/views/upright/dashboards/_uptime_bars.html.erb +17 -0
- data/app/views/upright/dashboards/_uptime_probe_row.html.erb +22 -0
- data/app/views/upright/dashboards/uptimes/show.html.erb +17 -0
- data/app/views/upright/jobs/show.html.erb +1 -0
- data/app/views/upright/probe_results/_pagination.html.erb +19 -0
- data/app/views/upright/probe_results/index.html.erb +72 -0
- data/app/views/upright/prometheus_proxy/show.html.erb +1 -0
- data/app/views/upright/sessions/new.html.erb +6 -0
- data/app/views/upright/sites/index.html.erb +22 -0
- data/config/brakeman.ignore +39 -0
- data/config/ci.rb +7 -0
- data/config/importmap.rb +18 -0
- data/config/routes.rb +41 -0
- data/db/migrate/20250114000001_create_upright_probe_results.rb +19 -0
- data/lib/generators/upright/install/install_generator.rb +83 -0
- data/lib/generators/upright/install/templates/alertmanager.yml +14 -0
- data/lib/generators/upright/install/templates/deploy.yml +118 -0
- data/lib/generators/upright/install/templates/development_alertmanager.yml +11 -0
- data/lib/generators/upright/install/templates/development_prometheus.yml +12 -0
- data/lib/generators/upright/install/templates/docker-compose.yml +38 -0
- data/lib/generators/upright/install/templates/http_probes.yml +14 -0
- data/lib/generators/upright/install/templates/omniauth.rb +8 -0
- data/lib/generators/upright/install/templates/otel_collector.yml +24 -0
- data/lib/generators/upright/install/templates/prometheus.yml +10 -0
- data/lib/generators/upright/install/templates/puma.rb +40 -0
- data/lib/generators/upright/install/templates/sites.yml +26 -0
- data/lib/generators/upright/install/templates/smtp_probes.yml +9 -0
- data/lib/generators/upright/install/templates/upright.rb +21 -0
- data/lib/generators/upright/install/templates/upright.rules.yml +256 -0
- data/lib/generators/upright/playwright_probe/playwright_probe_generator.rb +30 -0
- data/lib/generators/upright/playwright_probe/templates/authenticator.rb.tt +14 -0
- data/lib/generators/upright/playwright_probe/templates/probe.rb.tt +14 -0
- data/lib/omniauth/strategies/static_credentials.rb +57 -0
- data/lib/tasks/upright_tasks.rake +4 -0
- data/lib/upright/configuration.rb +106 -0
- data/lib/upright/engine.rb +157 -0
- data/lib/upright/metrics.rb +62 -0
- data/lib/upright/playwright/collect_performance_metrics.js +36 -0
- data/lib/upright/site.rb +49 -0
- data/lib/upright/tracing.rb +49 -0
- data/lib/upright/version.rb +3 -0
- data/lib/upright.rb +68 -0
- 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
|