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:
@@ -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 · 2–40 chars · a–z, 0–9, 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>
|
||||
Reference in New Issue
Block a user