* fix: server search sort by * fix: medal server listing upgrade button * fix: bottom pagination spacing
965 lines
29 KiB
Vue
965 lines
29 KiB
Vue
<template>
|
|
<div
|
|
data-pyro-server-list-root
|
|
class="experimental-styles-within relative mx-auto mb-6 flex w-full flex-col p-6"
|
|
:class="serverList.length ? 'min-h-screen' : 'min-h-[calc(100vh-4.5rem)]'"
|
|
>
|
|
<ServersGuestPlanModal
|
|
ref="guestPlanModal"
|
|
:available-products="pyroProducts"
|
|
:currency="selectedCurrency"
|
|
:logged-in="loggedIn"
|
|
@continue="handleGuestPlanContinue"
|
|
/>
|
|
<ModrinthServersPurchaseModal
|
|
v-if="customer && paymentMethods && regions"
|
|
ref="purchaseModal"
|
|
:publishable-key="props.stripePublishableKey"
|
|
:initiate-payment="
|
|
async (body) => await client.labrinth.billing_internal.initiatePayment(body)
|
|
"
|
|
:available-products="pyroProducts"
|
|
:on-error="handleError"
|
|
:customer="customer"
|
|
:payment-methods="paymentMethods"
|
|
:currency="selectedCurrency"
|
|
:pings="regionPings"
|
|
:regions="regions"
|
|
:refresh-payment-methods="fetchPaymentData"
|
|
:fetch-stock="fetchStock"
|
|
:affiliate-code="affiliateCode"
|
|
plan-stage
|
|
@purchase-success="handlePurchaseSuccess"
|
|
@hide="clearPurchaseIntent"
|
|
/>
|
|
<ServersUpgradeModalWrapper
|
|
v-if="loggedIn"
|
|
ref="medalUpgradeModal"
|
|
:stripe-publishable-key="props.stripePublishableKey"
|
|
:site-url="props.siteUrl"
|
|
:products="props.products"
|
|
/>
|
|
<ResubscribeModal ref="resubscribeModal" @resubscribe="handleResubscribeConfirm" />
|
|
|
|
<div
|
|
v-if="hasError"
|
|
class="mx-auto flex h-full min-h-[calc(100vh-4rem)] flex-col items-center justify-center gap-4 text-left"
|
|
>
|
|
<div class="flex max-w-lg flex-col items-center rounded-3xl bg-bg-raised p-6 shadow-xl">
|
|
<div class="flex flex-col items-center text-center">
|
|
<div class="flex flex-col items-center gap-4">
|
|
<div class="grid place-content-center rounded-full bg-bg-blue p-4">
|
|
<HammerIcon class="size-12 text-blue" />
|
|
</div>
|
|
<h1 class="m-0 w-fit text-3xl font-bold">{{ formatMessage(messages.errorTitle) }}</h1>
|
|
</div>
|
|
<p class="text-lg text-secondary">{{ formatMessage(messages.errorDescription) }}</p>
|
|
<ul class="m-0 list-disc space-y-4 p-0 pl-4 text-left text-sm leading-[170%]">
|
|
<li>{{ formatMessage(messages.errorAlertNotice) }}</li>
|
|
<li>
|
|
<IntlFormatted :message-id="messages.errorQueueNotice">
|
|
<template #warning="{ children }">
|
|
<span class="font-medium text-contrast"><component :is="() => children" /></span>
|
|
</template>
|
|
</IntlFormatted>
|
|
</li>
|
|
<li>{{ formatMessage(messages.errorSupportNotice) }}</li>
|
|
|
|
<li v-if="fetchError" class="text-red">
|
|
<p>{{ formatMessage(messages.errorDetails) }}</p>
|
|
<CopyCode
|
|
:text="(fetchError as ModrinthServersFetchError).message || 'Unknown error'"
|
|
:copyable="false"
|
|
:selectable="false"
|
|
:language="'json'"
|
|
/>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
<ButtonStyled size="large" type="standard" color="brand">
|
|
<AutoLink class="mt-6 !w-full" to="https://support.modrinth.com">{{
|
|
formatMessage(messages.contactSupportButton)
|
|
}}</AutoLink>
|
|
</ButtonStyled>
|
|
<ButtonStyled size="large" @click="() => router.go(0)">
|
|
<button class="mt-3 !w-full">{{ formatMessage(messages.reloadButton) }}</button>
|
|
</ButtonStyled>
|
|
</div>
|
|
</div>
|
|
|
|
<template v-else>
|
|
<div
|
|
class="relative flex h-fit w-full flex-col mb-4 items-center justify-between md:flex-row"
|
|
>
|
|
<h1 class="w-full text-2xl m-0 font-extrabold text-contrast">
|
|
{{ formatMessage(messages.serversTitle) }}
|
|
</h1>
|
|
<div class="flex w-full flex-row items-center justify-end gap-2 md:mb-0">
|
|
<StyledInput
|
|
id="search"
|
|
v-model="searchInput"
|
|
:icon="SearchIcon"
|
|
type="search"
|
|
name="search"
|
|
autocomplete="off"
|
|
:disabled="showServersListLoading"
|
|
:placeholder="formatMessage(messages.searchPlaceholder, { count: filteredData.length })"
|
|
wrapper-class="w-full md:w-72"
|
|
/>
|
|
<ButtonStyled type="standard" color="brand">
|
|
<button @click="openPurchaseModal">
|
|
<PlusIcon />
|
|
{{ formatMessage(messages.newServerButton) }}
|
|
</button>
|
|
</ButtonStyled>
|
|
</div>
|
|
</div>
|
|
|
|
<Transition name="fade" mode="out-in">
|
|
<div v-if="showServersListLoading" key="loading" class="flex flex-col gap-3">
|
|
<div
|
|
v-for="i in 3"
|
|
:key="i"
|
|
class="flex animate-pulse flex-row items-center gap-4 overflow-x-hidden rounded-2xl border-[1px] border-solid border-button-bg bg-bg-raised p-4"
|
|
>
|
|
<div class="size-16 rounded-xl bg-button-bg"></div>
|
|
<div class="flex flex-1 flex-col gap-2">
|
|
<div class="h-6 w-48 rounded bg-button-bg"></div>
|
|
<div class="h-4 w-64 rounded bg-button-bg opacity-75"></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div
|
|
v-else-if="serverList.length === 0 && !isPollingForNewServers"
|
|
key="empty"
|
|
class="flex h-full flex-col items-center justify-center gap-8 grow max-h-[1100px]"
|
|
>
|
|
<ServerListEmpty
|
|
:logged-in="loggedIn"
|
|
@click-new-server="openPurchaseModal"
|
|
@click-sign-in="handleSignIn"
|
|
/>
|
|
</div>
|
|
|
|
<div v-else key="list">
|
|
<Transition
|
|
enter-active-class="transition-all duration-300 ease-out"
|
|
enter-from-class="opacity-0 max-h-0"
|
|
enter-to-class="opacity-100 max-h-20"
|
|
leave-active-class="transition-all duration-200 ease-in"
|
|
leave-from-class="opacity-100 max-h-20"
|
|
leave-to-class="opacity-0 max-h-0"
|
|
>
|
|
<div
|
|
v-if="showPollingForNewServers"
|
|
class="bg-brand/10 my-4 flex items-center justify-center gap-2 rounded-full px-4 py-2 text-sm text-brand"
|
|
>
|
|
<LoaderCircleIcon class="size-4 animate-spin" />
|
|
<span>{{ formatMessage(messages.checkingForNewServers) }}</span>
|
|
</div>
|
|
</Transition>
|
|
|
|
<TransitionGroup
|
|
v-if="filteredData.length > 0 || isPollingForNewServers"
|
|
name="list"
|
|
tag="ul"
|
|
class="m-0 flex flex-col gap-3 p-0"
|
|
>
|
|
<MedalServerListing
|
|
v-for="server in filteredData.filter((s) => s.is_medal)"
|
|
:key="server.server_id"
|
|
v-bind="server"
|
|
@upgrade="openMedalUpgradeModal"
|
|
/>
|
|
<ServerListing
|
|
v-for="server in filteredData.filter((s) => !s.is_medal)"
|
|
:key="server.server_id"
|
|
v-bind="server"
|
|
:cancellation-date="serverBillingMap.get(server.server_id)?.cancellationDate"
|
|
:is-provisioning="serverBillingMap.get(server.server_id)?.isProvisioning"
|
|
:on-resubscribe="serverBillingMap.get(server.server_id)?.onResubscribe"
|
|
:on-download-backup="serverBillingMap.get(server.server_id)?.onDownloadBackup"
|
|
/>
|
|
</TransitionGroup>
|
|
<div v-else>{{ formatMessage(messages.noServersFound) }}</div>
|
|
</div>
|
|
</Transition>
|
|
</template>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import type { Archon, Labrinth } from '@modrinth/api-client'
|
|
import { HammerIcon, LoaderCircleIcon, PlusIcon, SearchIcon } from '@modrinth/assets'
|
|
import {
|
|
AutoLink,
|
|
ButtonStyled,
|
|
CopyCode,
|
|
defineMessages,
|
|
injectAuth,
|
|
injectModrinthClient,
|
|
injectNotificationManager,
|
|
IntlFormatted,
|
|
ModrinthServersPurchaseModal,
|
|
ResubscribeModal,
|
|
ServerListEmpty,
|
|
ServersGuestPlanModal,
|
|
StyledInput,
|
|
useServerBackupDownload,
|
|
useVIntl,
|
|
} from '@modrinth/ui'
|
|
import type { ModrinthServersFetchError } from '@modrinth/utils'
|
|
import { useQuery, useQueryClient } from '@tanstack/vue-query'
|
|
import { useIntervalFn } from '@vueuse/core'
|
|
import dayjs from 'dayjs'
|
|
import Fuse from 'fuse.js'
|
|
import type Stripe from 'stripe'
|
|
import { type ComponentPublicInstance, computed, ref, watch } from 'vue'
|
|
import { useRoute, useRouter } from 'vue-router'
|
|
|
|
import ServersUpgradeModalWrapper from '#ui/components/billing/ServersUpgradeModalWrapper.vue'
|
|
import MedalServerListing from '#ui/components/servers/marketing/MedalServerListing.vue'
|
|
import ServerListing from '#ui/components/servers/ServerListing.vue'
|
|
import { createHostingPurchaseIntentContext, provideHostingPurchaseIntent } from '#ui/providers'
|
|
|
|
const props = defineProps<{
|
|
stripePublishableKey: string
|
|
siteUrl?: string
|
|
products: Labrinth.Billing.Internal.Product[]
|
|
}>()
|
|
|
|
const router = useRouter()
|
|
const route = useRoute()
|
|
const auth = injectAuth()
|
|
const client = injectModrinthClient()
|
|
const loggedIn = computed(() => !!auth.user.value)
|
|
const { formatMessage } = useVIntl()
|
|
|
|
const messages = defineMessages({
|
|
errorTitle: { id: 'servers.manage.error.title', defaultMessage: 'Servers could not be loaded' },
|
|
errorDescription: {
|
|
id: 'servers.manage.error.description',
|
|
defaultMessage: 'We may have temporary issues with our servers.',
|
|
},
|
|
errorAlertNotice: {
|
|
id: 'servers.manage.error.alert-notice',
|
|
defaultMessage:
|
|
"Our systems automatically alert our team when there's an issue. We are already working on getting them back online.",
|
|
},
|
|
errorQueueNotice: {
|
|
id: 'servers.manage.error.queue-notice',
|
|
defaultMessage:
|
|
"If you recently purchased your Modrinth Hosting server, it is currently in a queue and will appear here as soon as it's ready. <warning>Do not attempt to purchase a new server.</warning>",
|
|
},
|
|
errorSupportNotice: {
|
|
id: 'servers.manage.error.support-notice',
|
|
defaultMessage:
|
|
'If you require personalized support regarding the status of your server, please contact Modrinth Support.',
|
|
},
|
|
errorDetails: { id: 'servers.manage.error.details', defaultMessage: 'Error details:' },
|
|
contactSupportButton: {
|
|
id: 'servers.manage.contact-support-button',
|
|
defaultMessage: 'Contact Modrinth Support',
|
|
},
|
|
reloadButton: { id: 'servers.manage.reload-button', defaultMessage: 'Reload' },
|
|
serversTitle: { id: 'servers.manage.servers-title', defaultMessage: 'Modrinth Hosting' },
|
|
searchPlaceholder: {
|
|
id: 'servers.manage.search-placeholder',
|
|
defaultMessage: 'Search {count} {count, plural, one {server} other {servers}}...',
|
|
},
|
|
newServerButton: { id: 'servers.manage.new-server-button', defaultMessage: 'New server' },
|
|
checkingForNewServers: {
|
|
id: 'servers.manage.checking-for-new-servers',
|
|
defaultMessage: 'Checking for new servers...',
|
|
},
|
|
noServersFound: { id: 'servers.manage.no-servers-found', defaultMessage: 'No servers found.' },
|
|
handleErrorTitle: {
|
|
id: 'servers.manage.handle-error.title',
|
|
defaultMessage: 'An error occurred',
|
|
},
|
|
purchaseUnavailableTitle: {
|
|
id: 'servers.manage.purchase-unavailable.title',
|
|
defaultMessage: 'Purchase unavailable',
|
|
},
|
|
purchaseUnavailableText: {
|
|
id: 'servers.manage.purchase-unavailable.text',
|
|
defaultMessage:
|
|
'Payment information is still loading. Opening checkout as soon as it is ready.',
|
|
},
|
|
resubscribeSubmittedTitle: {
|
|
id: 'servers.manage.resubscribe-submitted.title',
|
|
defaultMessage: 'Resubscription request submitted',
|
|
},
|
|
resubscribeSubmittedText: {
|
|
id: 'servers.manage.resubscribe-submitted.text',
|
|
defaultMessage:
|
|
'If the server is currently cancelled, it may take up to 10 minutes for another charge attempt to be made.',
|
|
},
|
|
resubscribeSuccessTitle: {
|
|
id: 'servers.manage.resubscribe-success.title',
|
|
defaultMessage: 'Success',
|
|
},
|
|
resubscribeSuccessText: {
|
|
id: 'servers.manage.resubscribe-success.text',
|
|
defaultMessage: 'Server subscription resubscribed successfully',
|
|
},
|
|
resubscribeErrorTitle: {
|
|
id: 'servers.manage.resubscribe-error.title',
|
|
defaultMessage: 'Error resubscribing',
|
|
},
|
|
resubscribeErrorText: {
|
|
id: 'servers.manage.resubscribe-error.text',
|
|
defaultMessage: 'An error occurred while resubscribing to your Modrinth server.',
|
|
},
|
|
})
|
|
|
|
const isPollingForNewServers = ref(false)
|
|
const showPollingForNewServers = ref(false)
|
|
let pollingShowTimeout: ReturnType<typeof setTimeout> | undefined
|
|
|
|
watch(isPollingForNewServers, (polling) => {
|
|
clearTimeout(pollingShowTimeout)
|
|
if (polling) {
|
|
pollingShowTimeout = setTimeout(() => {
|
|
showPollingForNewServers.value = isPollingForNewServers.value
|
|
}, 1500)
|
|
} else {
|
|
showPollingForNewServers.value = false
|
|
}
|
|
})
|
|
|
|
const pollingState = ref({
|
|
enabled: false,
|
|
count: 0,
|
|
initialServerIds: new Set<string>(),
|
|
})
|
|
|
|
function startNewServerPolling(initialServers: Archon.Servers.v0.Server[]) {
|
|
if (pollingState.value.enabled) return
|
|
isPollingForNewServers.value = true
|
|
pollingState.value = {
|
|
enabled: true,
|
|
count: 0,
|
|
initialServerIds: new Set(initialServers.map((s) => s.server_id)),
|
|
}
|
|
}
|
|
|
|
const guestPlanModal = ref<InstanceType<typeof ServersGuestPlanModal> | null>(null)
|
|
const purchaseModal = ref<InstanceType<typeof ModrinthServersPurchaseModal> | null>(null)
|
|
type UpgradeModalRef = ComponentPublicInstance<{ open: (id?: string) => void | Promise<void> }>
|
|
const medalUpgradeModal = ref<UpgradeModalRef | null>(null)
|
|
const resubscribeModal = ref<InstanceType<typeof ResubscribeModal> | null>(null)
|
|
const affiliateCode = ref<string | null>(null)
|
|
const selectedCurrency = ref<string>('USD')
|
|
const regionPings = ref<
|
|
{
|
|
region: string
|
|
ping: number
|
|
}[]
|
|
>([])
|
|
|
|
const pyroProducts = computed(() => {
|
|
return [...props.products]
|
|
.filter((p) => p?.metadata?.type === 'pyro' || p?.metadata?.type === 'medal')
|
|
.sort((a, b) => {
|
|
const aRam =
|
|
a?.metadata?.type === 'pyro' || a?.metadata?.type === 'medal' ? a.metadata.ram : 0
|
|
const bRam =
|
|
b?.metadata?.type === 'pyro' || b?.metadata?.type === 'medal' ? b.metadata.ram : 0
|
|
return aRam - bRam
|
|
})
|
|
})
|
|
|
|
const {
|
|
data: customer,
|
|
refetch: refetchCustomer,
|
|
isLoading: customerLoading,
|
|
} = useQuery({
|
|
queryKey: ['billing', 'customer'],
|
|
queryFn: () => client.labrinth.billing_internal.getCustomer() as Promise<Stripe.Customer>,
|
|
enabled: loggedIn,
|
|
})
|
|
|
|
const {
|
|
data: paymentMethods,
|
|
refetch: refetchPaymentMethods,
|
|
isLoading: paymentMethodsLoading,
|
|
} = useQuery({
|
|
queryKey: ['billing', 'payment-methods'],
|
|
queryFn: () =>
|
|
client.labrinth.billing_internal.getPaymentMethods() as Promise<Stripe.PaymentMethod[]>,
|
|
enabled: loggedIn,
|
|
})
|
|
|
|
const { data: regions, isLoading: regionsLoading } = useQuery({
|
|
queryKey: ['servers', 'regions'],
|
|
queryFn: () => client.archon.servers_v1.getRegions(),
|
|
enabled: loggedIn,
|
|
})
|
|
|
|
watch(
|
|
regions,
|
|
(newRegions) => {
|
|
regionPings.value = []
|
|
if (newRegions) {
|
|
newRegions.forEach((region) => {
|
|
runPingTest(region)
|
|
})
|
|
}
|
|
},
|
|
{ immediate: true },
|
|
)
|
|
|
|
async function fetchPaymentData() {
|
|
await Promise.all([refetchCustomer(), refetchPaymentMethods()])
|
|
}
|
|
|
|
async function fetchStock(
|
|
region: Archon.Servers.v1.Region,
|
|
request: Archon.Servers.v0.StockRequest,
|
|
): Promise<number> {
|
|
const result = await client.archon.servers_v0.checkStock(region.shortcode, request)
|
|
return result.available
|
|
}
|
|
|
|
const PING_COUNT = 20
|
|
const PING_INTERVAL = 200
|
|
const MAX_PING_TIME = 1000
|
|
|
|
function runPingTest(region: Archon.Servers.v1.Region, index = 1) {
|
|
if (index > 10) {
|
|
regionPings.value = regionPings.value.filter((entry) => entry.region !== region.shortcode)
|
|
regionPings.value.push({
|
|
region: region.shortcode,
|
|
ping: -1,
|
|
})
|
|
return
|
|
}
|
|
|
|
const wsUrl = `wss://${region.shortcode}${index}.${region.zone}/pingtest`
|
|
try {
|
|
const socket = new WebSocket(wsUrl)
|
|
const pings: number[] = []
|
|
let finalized = false
|
|
|
|
const finalize = (ping: number) => {
|
|
if (finalized) return
|
|
finalized = true
|
|
clearTimeout(connectTimeout)
|
|
regionPings.value = regionPings.value.filter((entry) => entry.region !== region.shortcode)
|
|
regionPings.value.push({
|
|
region: region.shortcode,
|
|
ping,
|
|
})
|
|
socket.close()
|
|
}
|
|
|
|
const retryNext = () => {
|
|
if (finalized) return
|
|
finalized = true
|
|
clearTimeout(connectTimeout)
|
|
socket.close()
|
|
runPingTest(region, index + 1)
|
|
}
|
|
|
|
// Prevent hangs where the socket never opens or errors.
|
|
const connectTimeout = setTimeout(() => {
|
|
retryNext()
|
|
}, 3000)
|
|
|
|
socket.onopen = () => {
|
|
clearTimeout(connectTimeout)
|
|
|
|
for (let i = 0; i < PING_COUNT; i++) {
|
|
setTimeout(() => {
|
|
socket.send(String(performance.now()))
|
|
}, i * PING_INTERVAL)
|
|
}
|
|
setTimeout(
|
|
() => {
|
|
const median =
|
|
pings.length > 0
|
|
? Math.round([...pings].sort((a, b) => a - b)[Math.floor(pings.length / 2)])
|
|
: -1
|
|
finalize(median)
|
|
},
|
|
PING_COUNT * PING_INTERVAL + MAX_PING_TIME,
|
|
)
|
|
}
|
|
|
|
socket.onmessage = (event) => {
|
|
const start = Number(event.data)
|
|
pings.push(performance.now() - start)
|
|
}
|
|
|
|
socket.onerror = () => {
|
|
retryNext()
|
|
}
|
|
} catch {
|
|
runPingTest(region, index + 1)
|
|
}
|
|
}
|
|
|
|
const {
|
|
data: serverResponse,
|
|
error: fetchError,
|
|
isPending: serversQueryPending,
|
|
} = useQuery({
|
|
queryKey: ['servers'],
|
|
queryFn: async () => {
|
|
const response = await client.archon.servers_v0.list({ limit: 100 })
|
|
|
|
// Fetch subscriptions for medal servers
|
|
const hasMedalServers = response.servers.some((s) => s.is_medal)
|
|
if (hasMedalServers) {
|
|
const subscriptions = await client.labrinth.billing_internal.getSubscriptions()
|
|
|
|
// Inject medal_expires into servers
|
|
for (const server of response.servers) {
|
|
if (server.is_medal) {
|
|
const sub = subscriptions.find((s) => s.metadata?.id === server.server_id)
|
|
if (sub) {
|
|
server.medal_expires = dayjs(sub.created).add(5, 'days').toISOString()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Check if new servers appeared (stop polling)
|
|
if (pollingState.value.enabled) {
|
|
pollingState.value.count++
|
|
const hasNewServer = response.servers.some(
|
|
(s) => !pollingState.value.initialServerIds.has(s.server_id),
|
|
)
|
|
if (hasNewServer) {
|
|
pollingState.value.enabled = false
|
|
isPollingForNewServers.value = false
|
|
|
|
router.replace({ query: {} })
|
|
} else if (pollingState.value.count >= 5) {
|
|
pollingState.value.enabled = false
|
|
isPollingForNewServers.value = false
|
|
}
|
|
}
|
|
|
|
return response
|
|
},
|
|
refetchInterval: computed(() => (pollingState.value.enabled ? 5000 : false)),
|
|
enabled: loggedIn,
|
|
})
|
|
|
|
const hasError = computed(() => loggedIn.value && !!fetchError.value)
|
|
|
|
/** Logged-in initial fetch: avoid treating "no data yet" as an empty server list. */
|
|
const showServersListLoading = computed(() => loggedIn.value && serversQueryPending.value)
|
|
|
|
const serverList = computed<Archon.Servers.v0.Server[]>(() => {
|
|
if (!loggedIn.value || !serverResponse.value) return []
|
|
return serverResponse.value.servers
|
|
})
|
|
|
|
const searchInput = ref('')
|
|
|
|
const fuse = computed(() => {
|
|
if (serverList.value.length === 0) return null
|
|
return new Fuse(serverList.value, {
|
|
keys: ['name', 'loader', 'mc_version', 'game', 'state'],
|
|
includeScore: true,
|
|
threshold: 0.4,
|
|
})
|
|
})
|
|
|
|
function isSetToCancel(server: Archon.Servers.v0.Server): boolean {
|
|
return (
|
|
server.status !== 'suspended' &&
|
|
Boolean(serverBillingMap.value.get(server.server_id)?.cancellationDate)
|
|
)
|
|
}
|
|
|
|
function getStatusPriority(server: Archon.Servers.v0.Server): number {
|
|
if (server.status === 'suspended') return 2
|
|
if (isSetToCancel(server)) return 1
|
|
return 0
|
|
}
|
|
|
|
function sortServers(array: Archon.Servers.v0.Server[]): Archon.Servers.v0.Server[] {
|
|
return array.slice().sort((a, b) => {
|
|
const priorityDiff = getStatusPriority(a) - getStatusPriority(b)
|
|
if (priorityDiff !== 0) return priorityDiff
|
|
|
|
const introDiff = Number(b.flows?.intro) - Number(a.flows?.intro)
|
|
if (introDiff !== 0) return introDiff
|
|
|
|
return (a.name || '').localeCompare(b.name || '')
|
|
})
|
|
}
|
|
|
|
// files expire 30 days after cancellation
|
|
function filesExpired(server: Archon.Servers.v0.Server): boolean {
|
|
if (server.status !== 'suspended' || server.suspension_reason !== 'cancelled') return false
|
|
const cancellationDate = serverBillingMap.value.get(server.server_id)?.cancellationDate
|
|
if (!cancellationDate) return false
|
|
const cancellation = new Date(cancellationDate)
|
|
const thirtyDaysLater = new Date(cancellation.getTime() + 30 * 24 * 60 * 60 * 1000)
|
|
return new Date() > thirtyDaysLater
|
|
}
|
|
|
|
const filteredData = computed<Archon.Servers.v0.Server[]>(() => {
|
|
const base = !searchInput.value.trim()
|
|
? sortServers(serverList.value)
|
|
: fuse.value
|
|
? sortServers(fuse.value.search(searchInput.value).map((result) => result.item))
|
|
: []
|
|
return base.filter((server) => !filesExpired(server))
|
|
})
|
|
|
|
// Start polling only after initial data is available so the baseline is correct
|
|
watch(serverResponse, (response) => {
|
|
if (
|
|
route.query.redirect_status === 'succeeded' &&
|
|
response &&
|
|
!pollingState.value.enabled &&
|
|
pollingState.value.count === 0
|
|
) {
|
|
startNewServerPolling(response.servers)
|
|
}
|
|
})
|
|
|
|
const { addNotification } = injectNotificationManager()
|
|
const queryClient = useQueryClient()
|
|
const { getLatestBackupDownload } = useServerBackupDownload()
|
|
|
|
function handlePurchaseSuccess() {
|
|
startNewServerPolling(serverResponse.value?.servers ?? [])
|
|
void Promise.all([
|
|
queryClient.invalidateQueries({ queryKey: ['servers'] }),
|
|
queryClient.invalidateQueries({ queryKey: ['servers', 'v1'] }),
|
|
])
|
|
}
|
|
|
|
watch(
|
|
() => auth.user.value,
|
|
(user, previousUser) => {
|
|
if (user || !previousUser) return
|
|
isPollingForNewServers.value = false
|
|
pollingState.value = {
|
|
enabled: false,
|
|
count: 0,
|
|
initialServerIds: new Set(),
|
|
}
|
|
void Promise.all([
|
|
queryClient.resetQueries({ queryKey: ['billing'] }),
|
|
queryClient.resetQueries({ queryKey: ['servers'] }),
|
|
])
|
|
},
|
|
)
|
|
|
|
const canOpenPurchaseModal = computed(() => {
|
|
return (
|
|
Boolean(props.stripePublishableKey) &&
|
|
Boolean(customer.value) &&
|
|
paymentMethods.value !== undefined &&
|
|
Boolean(regions.value) &&
|
|
!customerLoading.value &&
|
|
!paymentMethodsLoading.value &&
|
|
!regionsLoading.value
|
|
)
|
|
})
|
|
|
|
function handleError(err: unknown) {
|
|
const error = err as Error & { data?: { description?: string } }
|
|
addNotification({
|
|
title: formatMessage(messages.handleErrorTitle),
|
|
type: 'error',
|
|
text: error?.message ?? error?.data?.description ?? String(err),
|
|
})
|
|
}
|
|
|
|
function handleSignIn() {
|
|
void auth.requestSignIn('/hosting/manage')
|
|
}
|
|
|
|
const hostingPurchaseIntent = createHostingPurchaseIntentContext({
|
|
authRequestSignIn: auth.requestSignIn,
|
|
signInRedirectPath: '/hosting/manage',
|
|
intentSource: 'hosting-manage',
|
|
loggedIn,
|
|
availableProducts: pyroProducts,
|
|
canOpenCheckout: canOpenPurchaseModal,
|
|
guestPlanModal,
|
|
checkoutModal: purchaseModal,
|
|
onCheckoutPending: () => {
|
|
addNotification({
|
|
title: formatMessage(messages.purchaseUnavailableTitle),
|
|
text: formatMessage(messages.purchaseUnavailableText),
|
|
type: 'info',
|
|
})
|
|
},
|
|
})
|
|
provideHostingPurchaseIntent(hostingPurchaseIntent)
|
|
|
|
const { openPurchaseModal, handleGuestPlanContinue, clearPurchaseIntent } = hostingPurchaseIntent
|
|
|
|
function openMedalUpgradeModal(serverId: string) {
|
|
medalUpgradeModal.value?.open(serverId)
|
|
}
|
|
|
|
const { data: subscriptions } = useQuery({
|
|
queryKey: ['billing', 'subscriptions'],
|
|
queryFn: () => client.labrinth.billing_internal.getSubscriptions(),
|
|
enabled: loggedIn,
|
|
})
|
|
|
|
const { data: charges } = useQuery({
|
|
queryKey: ['billing', 'payments'],
|
|
queryFn: () => client.labrinth.billing_internal.getPayments(),
|
|
enabled: loggedIn,
|
|
})
|
|
|
|
const CHARGE_POLL_INTERVAL_MS = 20_000
|
|
|
|
const hasProvisioningSubscription = computed(() => {
|
|
if (!subscriptions.value || !charges.value) return false
|
|
return subscriptions.value
|
|
.filter((s) => s?.metadata?.type === 'pyro')
|
|
.some((sub) => {
|
|
if (sub.status !== 'unprovisioned') return false
|
|
const charge = charges.value?.find((c) => c.subscription_id === sub.id)
|
|
return charge?.status === 'processing' || charge?.status === 'open'
|
|
})
|
|
})
|
|
|
|
const { pause: pauseChargePoll, resume: resumeChargePoll } = useIntervalFn(
|
|
() => {
|
|
queryClient.invalidateQueries({ queryKey: ['billing', 'payments'] })
|
|
queryClient.invalidateQueries({ queryKey: ['billing', 'subscriptions'] })
|
|
queryClient.invalidateQueries({ queryKey: ['servers'] })
|
|
},
|
|
CHARGE_POLL_INTERVAL_MS,
|
|
{ immediate: false },
|
|
)
|
|
|
|
watch(
|
|
hasProvisioningSubscription,
|
|
(isProvisioning) => {
|
|
if (isProvisioning) {
|
|
resumeChargePoll()
|
|
} else {
|
|
pauseChargePoll()
|
|
}
|
|
},
|
|
{ immediate: true },
|
|
)
|
|
|
|
const { data: serverFullList } = useQuery({
|
|
queryKey: ['servers', 'v1'],
|
|
queryFn: () => client.archon.servers_v1.list(),
|
|
enabled: loggedIn,
|
|
})
|
|
|
|
type ServerBillingInfo = {
|
|
cancellationDate?: string | null
|
|
isProvisioning?: boolean
|
|
onResubscribe?: () => void
|
|
onDownloadBackup?: (() => void) | null
|
|
}
|
|
|
|
type ResubscribeRequest = {
|
|
subscriptionId: string
|
|
wasSuspended: boolean
|
|
}
|
|
|
|
function getProductFromPriceId(priceId: string | null | undefined) {
|
|
if (!priceId) return null
|
|
|
|
return (
|
|
pyroProducts.value.find((product) => product.prices.some((price) => price.id === priceId)) ??
|
|
null
|
|
)
|
|
}
|
|
|
|
function getPlanName(product: Labrinth.Billing.Internal.Product | null): string {
|
|
if (!product) return 'Medium plan'
|
|
if (product.metadata.type !== 'pyro' && product.metadata.type !== 'medal') return 'Medium plan'
|
|
|
|
switch (product.metadata.ram) {
|
|
case 4096:
|
|
return 'Small plan'
|
|
case 6144:
|
|
return 'Medium plan'
|
|
case 8192:
|
|
return 'Large plan'
|
|
default:
|
|
return 'Custom plan'
|
|
}
|
|
}
|
|
|
|
function getRamGb(product: Labrinth.Billing.Internal.Product | null): number | undefined {
|
|
if (!product) return undefined
|
|
if (product.metadata.type !== 'pyro' && product.metadata.type !== 'medal') return undefined
|
|
|
|
return product.metadata.ram / 1024
|
|
}
|
|
|
|
function getStorageGb(product: Labrinth.Billing.Internal.Product | null): number | undefined {
|
|
if (!product) return undefined
|
|
if (product.metadata.type !== 'pyro' && product.metadata.type !== 'medal') return undefined
|
|
|
|
return product.metadata.storage / 1024
|
|
}
|
|
|
|
function getSharedCpus(product: Labrinth.Billing.Internal.Product | null): number | undefined {
|
|
if (!product) return undefined
|
|
if (product.metadata.type !== 'pyro' && product.metadata.type !== 'medal') return undefined
|
|
|
|
return product.metadata.cpu / 2
|
|
}
|
|
|
|
function getRecurringPrice(
|
|
product: Labrinth.Billing.Internal.Product | null,
|
|
interval: Labrinth.Billing.Internal.PriceDuration,
|
|
preferredCurrency?: string,
|
|
): { amount: number; currencyCode: string } | null {
|
|
if (!product) return null
|
|
|
|
const recurringPrices = product.prices.filter((price) => price.prices.type === 'recurring')
|
|
const preferredPrice = preferredCurrency
|
|
? recurringPrices.find((price) => price.currency_code === preferredCurrency)
|
|
: undefined
|
|
const usdPrice = recurringPrices.find((price) => price.currency_code === 'USD')
|
|
const selectedPrice = preferredPrice ?? usdPrice ?? recurringPrices[0]
|
|
|
|
if (!selectedPrice || selectedPrice.prices.type !== 'recurring') return null
|
|
|
|
return {
|
|
amount: selectedPrice.prices.intervals[interval],
|
|
currencyCode: selectedPrice.currency_code,
|
|
}
|
|
}
|
|
|
|
function openResubscribeModal(
|
|
serverId: string,
|
|
subscription: Labrinth.Billing.Internal.UserSubscription,
|
|
charge?: Labrinth.Billing.Internal.Charge | null,
|
|
) {
|
|
const displayInterval = charge?.subscription_interval ?? subscription.interval
|
|
const displayPriceId = charge?.price_id ?? subscription.price_id
|
|
const product = getProductFromPriceId(displayPriceId)
|
|
const fallbackPrice = getRecurringPrice(product, displayInterval, charge?.currency_code)
|
|
|
|
resubscribeModal.value?.show({
|
|
subscriptionId: subscription.id,
|
|
wasSuspended: !!charge?.due && dayjs(charge.due).isBefore(dayjs()),
|
|
serverName:
|
|
serverList.value.find((server) => server.server_id === serverId)?.name ?? 'this server',
|
|
planName: getPlanName(product),
|
|
ramGb: getRamGb(product),
|
|
storageGb: getStorageGb(product),
|
|
sharedCpus: getSharedCpus(product),
|
|
priceCents: charge?.amount ?? fallbackPrice?.amount,
|
|
currencyCode: charge?.currency_code ?? fallbackPrice?.currencyCode,
|
|
interval: displayInterval,
|
|
nextChargeDate: charge?.due,
|
|
})
|
|
}
|
|
|
|
async function handleResubscribeConfirm({ subscriptionId, wasSuspended }: ResubscribeRequest) {
|
|
try {
|
|
await client.labrinth.billing_internal.editSubscription(subscriptionId, {
|
|
cancelled: false,
|
|
})
|
|
await Promise.all([
|
|
queryClient.invalidateQueries({ queryKey: ['billing'] }),
|
|
queryClient.invalidateQueries({ queryKey: ['servers'] }),
|
|
])
|
|
if (wasSuspended) {
|
|
addNotification({
|
|
title: formatMessage(messages.resubscribeSubmittedTitle),
|
|
text: formatMessage(messages.resubscribeSubmittedText),
|
|
type: 'success',
|
|
})
|
|
} else {
|
|
addNotification({
|
|
title: formatMessage(messages.resubscribeSuccessTitle),
|
|
text: formatMessage(messages.resubscribeSuccessText),
|
|
type: 'success',
|
|
})
|
|
}
|
|
} catch {
|
|
addNotification({
|
|
title: formatMessage(messages.resubscribeErrorTitle),
|
|
text: formatMessage(messages.resubscribeErrorText),
|
|
type: 'error',
|
|
})
|
|
}
|
|
}
|
|
|
|
const serverBillingMap = computed(() => {
|
|
const map = new Map<string, ServerBillingInfo>()
|
|
if (!subscriptions.value || !charges.value) return map
|
|
|
|
const pyroSubs = subscriptions.value.filter((s) => s?.metadata?.type === 'pyro')
|
|
for (const sub of pyroSubs) {
|
|
const serverId = (sub.metadata as { id?: string })?.id
|
|
if (!serverId) continue
|
|
|
|
const charge = charges.value.find(
|
|
(c) => c.subscription_id === sub.id && c.status !== 'succeeded',
|
|
)
|
|
|
|
const info: ServerBillingInfo = {
|
|
isProvisioning:
|
|
sub.status === 'unprovisioned' &&
|
|
(charge?.status === 'processing' || charge?.status === 'open'),
|
|
}
|
|
|
|
info.onDownloadBackup = getLatestBackupDownload(serverId, serverFullList.value)
|
|
|
|
if (charge?.status === 'cancelled') {
|
|
info.cancellationDate = charge.due
|
|
|
|
info.onResubscribe = () => openResubscribeModal(serverId, sub, charge)
|
|
}
|
|
|
|
map.set(serverId, info)
|
|
}
|
|
|
|
return map
|
|
})
|
|
</script>
|
|
|
|
<style scoped>
|
|
.fade-enter-active,
|
|
.fade-leave-active {
|
|
transition:
|
|
opacity 300ms ease-in-out,
|
|
transform 300ms ease-in-out;
|
|
}
|
|
|
|
.fade-enter-from,
|
|
.fade-leave-to {
|
|
opacity: 0;
|
|
transform: scale(0.98);
|
|
}
|
|
|
|
.list-enter-active,
|
|
.list-leave-active {
|
|
transition: all 200ms ease-in-out;
|
|
}
|
|
|
|
.list-enter-from {
|
|
opacity: 0;
|
|
transform: translateY(-10px);
|
|
}
|
|
|
|
.list-leave-to {
|
|
opacity: 0;
|
|
transform: translateY(10px);
|
|
}
|
|
|
|
.list-move {
|
|
transition: transform 200ms ease-in-out;
|
|
}
|
|
</style>
|