181 lines
3.9 KiB
Vue
181 lines
3.9 KiB
Vue
|
|
<template>
|
||
|
|
<div class="page">
|
||
|
|
<section class="card hero glass">
|
||
|
|
<div>
|
||
|
|
<p class="eyebrow">Atlas</p>
|
||
|
|
<h1>Onboarding</h1>
|
||
|
|
<p class="lede">Use your request code to view status and next steps.</p>
|
||
|
|
</div>
|
||
|
|
</section>
|
||
|
|
|
||
|
|
<section class="card module">
|
||
|
|
<div class="module-head">
|
||
|
|
<h2>Request Code</h2>
|
||
|
|
<span class="pill mono" :class="status ? 'pill-ok' : 'pill-warn'">
|
||
|
|
{{ status || "unknown" }}
|
||
|
|
</span>
|
||
|
|
</div>
|
||
|
|
|
||
|
|
<div class="status-form">
|
||
|
|
<input v-model="requestCode" class="input mono" type="text" placeholder="username~XXXXXXXXXX" :disabled="loading" />
|
||
|
|
<button class="primary" type="button" @click="check" :disabled="loading || !requestCode.trim()">
|
||
|
|
{{ loading ? "Checking..." : "Check" }}
|
||
|
|
</button>
|
||
|
|
</div>
|
||
|
|
|
||
|
|
<div v-if="status === 'approved'" class="steps">
|
||
|
|
<h3>Next steps</h3>
|
||
|
|
<ol>
|
||
|
|
<li>Log in at <a href="https://cloud.bstein.dev" target="_blank" rel="noreferrer">cloud.bstein.dev</a>.</li>
|
||
|
|
<li>Use your Keycloak username/password to access services.</li>
|
||
|
|
<li>If something doesn't work, contact the Atlas admin.</li>
|
||
|
|
</ol>
|
||
|
|
</div>
|
||
|
|
|
||
|
|
<div v-if="status === 'denied'" class="steps">
|
||
|
|
<h3>Denied</h3>
|
||
|
|
<p class="muted">This request was denied. Contact the Atlas admin if you think this is a mistake.</p>
|
||
|
|
</div>
|
||
|
|
|
||
|
|
<div v-if="error" class="error-box">
|
||
|
|
<div class="mono">{{ error }}</div>
|
||
|
|
</div>
|
||
|
|
</section>
|
||
|
|
</div>
|
||
|
|
</template>
|
||
|
|
|
||
|
|
<script setup>
|
||
|
|
import { onMounted, ref } from "vue";
|
||
|
|
import { useRoute } from "vue-router";
|
||
|
|
|
||
|
|
const route = useRoute();
|
||
|
|
|
||
|
|
const requestCode = ref("");
|
||
|
|
const status = ref("");
|
||
|
|
const loading = ref(false);
|
||
|
|
const error = ref("");
|
||
|
|
|
||
|
|
async function check() {
|
||
|
|
if (loading.value) return;
|
||
|
|
error.value = "";
|
||
|
|
loading.value = true;
|
||
|
|
try {
|
||
|
|
const resp = await fetch("/api/access/request/status", {
|
||
|
|
method: "POST",
|
||
|
|
headers: { "Content-Type": "application/json" },
|
||
|
|
body: JSON.stringify({ request_code: requestCode.value.trim() }),
|
||
|
|
});
|
||
|
|
const data = await resp.json().catch(() => ({}));
|
||
|
|
if (!resp.ok) throw new Error(data.error || resp.statusText || `status ${resp.status}`);
|
||
|
|
status.value = data.status || "unknown";
|
||
|
|
} catch (err) {
|
||
|
|
error.value = err.message || "Failed to check status";
|
||
|
|
} finally {
|
||
|
|
loading.value = false;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
onMounted(async () => {
|
||
|
|
const code = route.query.code || route.query.request_code || "";
|
||
|
|
if (typeof code === "string" && code.trim()) {
|
||
|
|
requestCode.value = code.trim();
|
||
|
|
await check();
|
||
|
|
}
|
||
|
|
});
|
||
|
|
</script>
|
||
|
|
|
||
|
|
<style scoped>
|
||
|
|
.page {
|
||
|
|
max-width: 960px;
|
||
|
|
margin: 0 auto;
|
||
|
|
padding: 32px 22px 72px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.hero {
|
||
|
|
margin-bottom: 12px;
|
||
|
|
padding: 18px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.eyebrow {
|
||
|
|
text-transform: uppercase;
|
||
|
|
letter-spacing: 0.08em;
|
||
|
|
color: var(--text-muted);
|
||
|
|
margin: 0 0 6px;
|
||
|
|
font-size: 13px;
|
||
|
|
}
|
||
|
|
|
||
|
|
h1 {
|
||
|
|
margin: 0 0 6px;
|
||
|
|
font-size: 32px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.lede {
|
||
|
|
margin: 0;
|
||
|
|
color: var(--text-muted);
|
||
|
|
max-width: 640px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.module {
|
||
|
|
padding: 18px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.module-head {
|
||
|
|
display: flex;
|
||
|
|
align-items: center;
|
||
|
|
justify-content: space-between;
|
||
|
|
gap: 18px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.status-form {
|
||
|
|
display: flex;
|
||
|
|
gap: 10px;
|
||
|
|
margin-top: 12px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.input {
|
||
|
|
flex: 1;
|
||
|
|
padding: 10px 12px;
|
||
|
|
border-radius: 10px;
|
||
|
|
border: 1px solid rgba(255, 255, 255, 0.14);
|
||
|
|
background: rgba(0, 0, 0, 0.25);
|
||
|
|
color: var(--text-primary);
|
||
|
|
}
|
||
|
|
|
||
|
|
button.primary {
|
||
|
|
background: linear-gradient(90deg, #4f8bff, #7dd0ff);
|
||
|
|
color: #0b1222;
|
||
|
|
padding: 10px 14px;
|
||
|
|
border: none;
|
||
|
|
border-radius: 10px;
|
||
|
|
cursor: pointer;
|
||
|
|
font-weight: 700;
|
||
|
|
}
|
||
|
|
|
||
|
|
.steps {
|
||
|
|
margin-top: 16px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.steps h3 {
|
||
|
|
margin: 0 0 8px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.steps ol {
|
||
|
|
margin: 0;
|
||
|
|
padding-left: 18px;
|
||
|
|
color: var(--text-muted);
|
||
|
|
}
|
||
|
|
|
||
|
|
.muted {
|
||
|
|
color: var(--text-muted);
|
||
|
|
}
|
||
|
|
|
||
|
|
.error-box {
|
||
|
|
margin-top: 12px;
|
||
|
|
border-radius: 12px;
|
||
|
|
border: 1px solid rgba(255, 87, 87, 0.5);
|
||
|
|
background: rgba(255, 87, 87, 0.06);
|
||
|
|
padding: 10px 12px;
|
||
|
|
}
|
||
|
|
</style>
|
||
|
|
|