refactor(operator): remove tenant creation (tenants are partner-owned)

Tenants always belong to a partner, so the operator must not create orphan (partnerless) tenants. Remove the NewTenantModal component, the POST /api/tenants proxy route, and the New-tenant button/modal from the tenants page. Tenant creation now happens only via the partner portal wizard (which forces partnerId).
This commit is contained in:
Ronni Baslund
2026-05-30 08:29:40 +02:00
parent 69197e11ae
commit 9c65a65bcd
3 changed files with 0 additions and 287 deletions
-265
View File
@@ -1,265 +0,0 @@
<script setup lang="ts">
// "New tenant" modal — opens from the Tenants page header. Collects the
// minimum needed to create a tenant (slug + name + plan + optional primary
// domain); billing details, status changes, and provisioning are edited from
// the tenant detail page after creation.
import type { Tenant, TenantPlan } from '~/types/tenant'
const props = defineProps<{ open: boolean }>()
const emit = defineEmits<{ close: []; created: [Tenant] }>()
const slug = ref('')
const name = ref('')
const plan = ref<TenantPlan>('mvp')
const domain = ref('')
const slugTouched = ref(false)
const error = ref<string | null>(null)
const busy = ref(false)
const PLANS: TenantPlan[] = ['mvp', 'pro', 'enterprise']
// Reset every time the modal opens so previous attempts don't linger.
watch(
() => props.open,
(v) => {
if (v) {
slug.value = ''
name.value = ''
plan.value = 'mvp'
domain.value = ''
slugTouched.value = false
error.value = null
busy.value = false
}
},
)
// Auto-derive slug from name until the operator types in the slug field
// themselves — once they touch slug we never overwrite it.
watch(name, (next) => {
if (slugTouched.value) return
slug.value = next
.toLowerCase()
.replace(/[^a-z0-9]+/g, '-')
.replace(/^-+|-+$/g, '')
.slice(0, 40)
})
onMounted(() => {
const onKey = (e: KeyboardEvent) => {
if (e.key === 'Escape' && props.open && !busy.value) emit('close')
}
document.addEventListener('keydown', onKey)
onBeforeUnmount(() => document.removeEventListener('keydown', onKey))
})
// Mirrors the platform-api CreateTenantDto validators so we don't fire a
// doomed POST. Backend re-checks on submit.
const slugValid = computed(() =>
/^[a-z0-9]([a-z0-9-]{0,38}[a-z0-9])?$/.test(slug.value.trim()),
)
const nameValid = computed(() => name.value.trim().length >= 2 && name.value.trim().length <= 120)
const domainValid = computed(() => {
const d = domain.value.trim()
return d.length === 0 || /^[a-z0-9.-]+\.[a-z]{2,}$/i.test(d)
})
const canSubmit = computed(
() => !busy.value && slugValid.value && nameValid.value && domainValid.value,
)
async function submit() {
if (!canSubmit.value) return
busy.value = true
error.value = null
const d = domain.value.trim()
try {
const created = await $fetch<Tenant>('/api/tenants', {
method: 'POST',
body: {
slug: slug.value.trim(),
name: name.value.trim(),
plan: plan.value,
...(d ? { domains: [d] } : {}),
},
})
emit('created', created)
} catch (e) {
const err = e as { data?: { data?: { message?: string }; message?: string; statusMessage?: string } }
error.value =
err.data?.data?.message ||
err.data?.message ||
err.data?.statusMessage ||
'Create failed'
} finally {
busy.value = false
}
}
</script>
<template>
<Teleport to="body">
<div v-if="open" class="backdrop" @click="!busy && emit('close')">
<div class="modal" role="dialog" aria-label="New tenant" @click.stop>
<header>
<div>
<Eyebrow>Customers</Eyebrow>
<h2>New tenant</h2>
</div>
<button class="x" type="button" aria-label="Close" :disabled="busy" @click="emit('close')">
<UiIcon name="x" :size="12" />
</button>
</header>
<div class="body">
<section>
<label class="label">Name</label>
<input v-model="name" type="text" placeholder="Acme ApS" :disabled="busy" />
</section>
<section>
<label class="label">Slug</label>
<input
v-model="slug"
type="text"
placeholder="acme"
:disabled="busy"
@input="slugTouched = true"
/>
<Mono dim>lowercase · 240 chars · az, 09, hyphens · stable URL key</Mono>
</section>
<section>
<label class="label">Plan</label>
<div class="seg three">
<button
v-for="p in PLANS"
:key="p"
type="button"
:class="{ on: plan === p }"
:disabled="busy"
@click="plan = p"
>{{ p }}</button>
</div>
</section>
<section>
<label class="label">Primary domain <span class="hint">(optional)</span></label>
<input v-model="domain" type="text" placeholder="acme.example.com" :disabled="busy" />
<Mono dim>can be added later from the tenant detail page</Mono>
</section>
<p v-if="error" class="err">{{ error }}</p>
</div>
<footer>
<UiButton variant="ghost" :disabled="busy" @click="emit('close')">Cancel</UiButton>
<UiButton variant="primary" :disabled="!canSubmit" @click="submit">
{{ busy ? 'Creating' : 'Create tenant' }}
</UiButton>
</footer>
</div>
</div>
</Teleport>
</template>
<style scoped>
.backdrop {
position: fixed; inset: 0;
background: rgba(0, 0, 0, 0.55);
display: flex; align-items: center; justify-content: center;
padding: 24px;
z-index: 180;
}
.modal {
width: 100%; max-width: 480px;
background: var(--elevated);
border: 1px solid var(--border);
border-radius: 10px;
box-shadow: 0 24px 80px rgba(0, 0, 0, 0.5);
display: flex; flex-direction: column;
overflow: hidden;
}
header {
padding: 16px 20px;
display: flex; justify-content: space-between; align-items: flex-start;
border-bottom: 1px solid var(--border);
}
h2 { margin: 4px 0 0 0; font-family: var(--font-display); font-weight: 600; font-size: 18px; }
.x {
width: 26px; height: 26px;
border: 0; border-radius: 6px; background: transparent;
color: var(--text-mute); cursor: pointer;
display: inline-flex; align-items: center; justify-content: center;
}
.x:hover:not(:disabled) { background: var(--surface); color: var(--text); }
.x:disabled { opacity: 0.4; cursor: not-allowed; }
.body { padding: 16px 20px; display: flex; flex-direction: column; gap: 14px; }
section { display: flex; flex-direction: column; gap: 6px; }
.label {
font-family: var(--font-mono);
font-size: 10px;
letter-spacing: 0.12em;
text-transform: uppercase;
color: var(--text-mute);
font-weight: 500;
}
.label .hint { color: var(--text-mute); font-weight: 400; text-transform: none; letter-spacing: 0; }
input {
height: 34px;
padding: 0 12px;
background: var(--input-bg);
border: 1px solid var(--border);
color: var(--text);
border-radius: 6px;
font-family: inherit;
font-size: 13px;
outline: 0;
}
input:focus { border-color: var(--border-hi); }
input:disabled { opacity: 0.6; }
.seg {
display: grid;
width: 100%;
gap: 2px;
padding: 2px;
background: var(--surface);
border: 1px solid var(--border);
border-radius: 7px;
box-sizing: border-box;
}
.seg.three { grid-template-columns: repeat(3, minmax(0, 1fr)); }
.seg button {
appearance: none;
border: 0;
background: transparent;
color: var(--text-dim);
font-family: inherit;
font-size: 11px;
font-weight: 500;
padding: 6px 10px;
border-radius: 5px;
cursor: pointer;
}
.seg button:hover:not(:disabled) { color: var(--text); }
.seg button.on { background: var(--text); color: var(--bg); }
.seg button:disabled { opacity: 0.5; cursor: not-allowed; }
.err {
margin: 0;
padding: 10px 12px;
background: rgba(240, 88, 88, 0.08);
border: 1px solid rgba(240, 88, 88, 0.24);
color: var(--bad);
border-radius: 6px;
font-size: 12px;
}
footer {
padding: 12px 20px;
display: flex;
justify-content: flex-end;
gap: 8px;
border-top: 1px solid var(--border);
background: var(--surface);
}
</style>