Files
cloudron-box/dashboard/src/views/SetupView.vue

252 lines
9.6 KiB
Vue
Raw Normal View History

<script setup>
import { ref, onMounted, useTemplateRef } from 'vue';
import { Spinner, Button, SingleSelect, FormGroup, TextInput } from 'pankow';
import { redirectIfNeeded } from '../utils.js';
import DomainsModel from '../models/DomainsModel.js';
import ProvisionModel from '../models/ProvisionModel.js';
import DomainProviderForm from '../components/DomainProviderForm.vue';
const provisionModel = ProvisionModel.create();
const ipProviders = [
{ name: 'Disabled', value: 'noop' },
{ name: 'Public IP', value: 'generic' },
{ name: 'Static IP Address', value: 'fixed' },
{ name: 'Network Interface', value: 'network-interface' }
];
const formError = ref({});
const busy = ref(false);
const ready = ref(false);
const waitingForDnsSetup = ref(false);
const progressMessage = ref('');
const taskMinutesActive = ref(0);
const domain = ref('');
const instanceId = ref('');
const setupToken = ref('');
const zoneName = ref('');
const provider = ref('');
const dnsConfig = ref(DomainsModel.createEmptyConfig());
const tlsProvider = ref('letsencrypt-prod-wildcard');
const showAdvanced = ref(false);
const ipv4Provider = ref('generic');
const ipv4Address = ref('');
const ipv4Interface = ref('');
const ipv6Provider = ref('generic');
const ipv6Address = ref('');
const ipv6Interface = ref('');
const form = useTemplateRef('form');
const isFormValid = ref(false);
function checkValidity() {
if (!provider.value) return false;
isFormValid.value = form.value.checkValidity();
}
async function waitForDnsSetup () {
waitingForDnsSetup.value = true;
formError.value = {};
const [error, result] = await provisionModel.status();
if (error) {
setTimeout(waitForDnsSetup, 5000);
return console.error(error);
}
if (!result.setup.active) {
if (!result.adminFqdn || result.setup.errorMessage) { // setup reset or errored. start over
formError.value.dnsWait = result.setup.errorMessage;
waitingForDnsSetup.value = false;
} else { // proceed to activation
window.location.href = 'https://' + result.adminFqdn + '/activation.html' + (window.location.search);
}
return;
}
progressMessage.value = result.setup.message;
taskMinutesActive.value = (new Date() - new Date(result.setup.startTime)) / 60000;
setTimeout(waitForDnsSetup, 5000);
}
async function onSubmit() {
if (!isFormValid.value) return;
busy.value = true;
formError.value = {};
const data = {
domainConfig: {
domain: domain.value,
zoneName: zoneName.value,
provider: provider.value,
config: DomainsModel.filterConfigForProvider(provider.value, dnsConfig.value),
tlsConfig: {
2025-04-02 15:31:43 +02:00
provider: tlsProvider.value.replace('-wildcard', ''),
wildcard: tlsProvider.value.indexOf('-wildcard') !== -1 ? true : false,
},
},
ipv4Config: {
provider: ipv4Provider.value,
ip: ipv4Provider.value === 'fixed' ? ipv4Address.value : '',
ifname: ipv4Provider.value === 'network-interface' ? ipv4Interface.value : '',
},
ipv6Config: {
provider: ipv6Provider.value,
ip: ipv6Provider.value === 'fixed' ? ipv6Address.value : '',
ifname: ipv6Provider.value === 'network-interface' ? ipv6Interface.value : '',
},
providerToken: instanceId.value,
setupToken: setupToken.value,
};
const [error] = await provisionModel.setup(data);
if (error) {
if (error.status === 422) {
if (provider.value === 'ami') {
formError.value.ami = error.body.message;
} else {
formError.value.setup = error.body.message;
}
} else {
formError.value.generic = error.body ? error.body.message : 'Internal error';
}
busy.value = false;
return;
}
waitForDnsSetup();
}
onMounted(async () => {
const search = decodeURIComponent(window.location.search).slice(1).split('&').map(function (item) { return item.split('='); }).reduce(function (o, k) { o[k[0]] = k[1]; return o; }, {});
const [error, result] = await provisionModel.status();
if (error) return console.error(error);
if (redirectIfNeeded(result, 'setup')) return; // redirected to some other view...
instanceId.value = search.instanceId;
setupToken.value = search.setupToken;
ready.value = true;
});
</script>
<template>
<div class="container" v-if="ready">
<Transition name="slide-fade" mode="out-in">
<div class="view" v-if="waitingForDnsSetup" style="text-align: center; max-width: unset;">
<Spinner class="pankow-spinner-large"/>
<h3>{{ progressMessage }} ...</h3>
<div>
Please wait while Cloudron is setting up the dashboard.
<br/>
<br/>
You can follow the logs on the server at <code class="clipboard hand" data-clipboard-text="/home/yellowtent/platformdata/logs/box.log" uib-tooltip="{{ clipboardDone ? 'Copied' : 'Click to copy' }}" tooltip-placement="right">/home/yellowtent/platformdata/logs/box.log</code>
<br/>
<br/>
<span v-show="taskMinutesActive >= 4">If setup appears stuck, it can be restarted by running <code>systemctl restart box</code> and reloading this page.</span>
</div>
</div>
<div class="view" v-else>
<h1>Cloudron Domain Setup</h1>
<div class="text-danger" v-if="formError.dnsWait">{{ formError.dnsWait }}</div>
<div class="text-danger" v-if="formError.setup">{{ formError.setup }}</div>
<div class="text-danger" v-if="formError.generic">{{ formError.generic }}</div>
<form ref="form" @submit.prevent="onSubmit()" @input="checkValidity()">
<fieldset :disabled="busy">
<input type="submit" style="display: none" :disabled="busy || !isFormValid"/>
<FormGroup>
<label for="domainInput">Domain <sup><a href="https://docs.cloudron.io/installation/#domain-setup" class="help" target="_blank"><i class="fa fa-question-circle"></i></a></sup></label>
<TextInput id="domainInput" v-model="domain" placeholder="example.com" required />
<div class="text-danger" v-show="domain.indexOf('my.') === 0 && domain.length > 3">Are you sure about this domain? The dashboard will be at <b>my.{{ domain }}</b></div>
<div style="padding-top: 6px;">Apps will be installed on subdomains of this domain. The dashboard will be available on the <b>my{{ domain ? `.${domain}` : '' }}</b> subdomain. You can add more domains later.</div>
</FormGroup>
<DomainProviderForm v-model:provider="provider" v-model:dns-config="dnsConfig" v-model:tls-provider="tlsProvider" :domain="domain" :show-advanced="showAdvanced" />
<div v-show="showAdvanced">
<FormGroup>
<label for="zoneNameInput">DNS Zone Name (Optional) <sup><a href="https://docs.cloudron.io/domains/#zone-name" class="help" target="_blank"><i class="fa fa-question-circle"></i></a></sup></label>
<TextInput id="zoneNameInput" v-model="zoneName" placeholder="Defaults to TLD" />
</FormGroup>
<!-- IPv4 provider -->
<FormGroup>
2025-05-06 11:45:49 +02:00
<label class="control-label">IPv4 Configuration <sup><a href="https://docs.cloudron.io/networking/#ip-configuration" class="help" target="_blank"><i class="fa fa-question-circle"></i></a></sup></label>
<SingleSelect v-model="ipv4Provider" :options="ipProviders" option-key="value" option-label="name" />
</FormGroup>
<!-- IPv4 Fixed -->
<FormGroup v-if="ipv4Provider === 'fixed'">
<label for="ipv4AddressInput">IPv4 Address</label>
<TextInput id="ipv4AddressInput" v-model="ipv4Address" required />
</FormGroup>
<!-- IPv4 Network Interface -->
<FormGroup v-if="ipv4Provider === 'network-interface'">
<label for="ipv4InterfaceInput">IPv4 Interface Name</label>
<TextInput id="ipv4InterfaceInput" v-model="ipv4Interface" required />
</FormGroup>
<!-- IPv6 provider -->
<FormGroup>
2025-05-06 11:45:49 +02:00
<label class="control-label">IPv6 Configuration <sup><a href="https://docs.cloudron.io/networking/#ip-configuration" class="help" target="_blank"><i class="fa fa-question-circle"></i></a></sup></label>
<SingleSelect v-model="ipv6Provider" :options="ipProviders" option-key="value" option-label="name" />
</FormGroup>
<!-- IPv6 Fixed -->
<FormGroup v-if="ipv6Provider === 'fixed'">
<label for="ipv6AddressInput">IPv6 Address</label>
<TextInput id="ipv6AddressInput" v-model="ipv6Address" required />
</FormGroup>
<!-- IPv6 Network Interface -->
<FormGroup v-if="ipv6Provider === 'network-interface'">
<label for="ipv6InterfaceInpt">IPv6 Interface Name</label>
<TextInput id="ipv6InterfaceInpt" v-model="ipv6Interface" required />
</FormGroup>
</div>
<div class="actionable" @click="showAdvanced = false" v-if="showAdvanced">Hide Advanced settings</div>
<div class="actionable" @click="showAdvanced = true" v-else>Advanced settings...</div>
</fieldset>
</form>
<Button @click="onSubmit()" style="margin-top: 12px" :disabled="busy || !isFormValid" :loading="busy">Next</Button>
<a href="/restore.html" style="margin-left: 10px;">Looking to restore?</a>
</div>
</Transition>
</div>
</template>
<style scoped>
.container {
display: flex;
align-items: center;
justify-content: center;
height: 100%;
overflow: auto;
}
h1 {
margin-top: 0;
}
.view {
width: 100%;
max-width: 500px;
max-height: 100%;
}
</style>