1
1
<script setup lang="ts">
2
2
import { storeToRefs } from ' pinia'
3
- import { ref , computed } from ' vue'
3
+ import { ref , computed , watch } from ' vue'
4
4
import { useToast } from ' vue-toastification'
5
5
6
6
import { useToastError } from ' @/composables/useToastError'
7
7
8
8
import { useAuthStore } from ' @/stores/auth'
9
- import { useTextareaInput } from ' @/composables/useTextareaInput'
10
- import { getGeneralSettings , getLoginSettings , updateGeneralSettings , updateLoginSettings , type GeneralSettingsResponseDTO as GeneralSettingsDTO , type LoginSettingsResponseDTO as LoginSettingsDTO } from ' @/api'
9
+ import { getMFAMethods , getGeneralSettings , getLoginSettings , updateGeneralSettings , updateLoginSettings , type GeneralSettingsResponseDTO as GeneralSettingsDTO , type LoginSettingsResponseDTO as LoginSettingsDTO } from ' @/api'
11
10
import { useApi } from ' @/composables/useApi'
12
11
import PageLoading from ' @/components/PageLoading.vue'
13
12
@@ -21,15 +20,24 @@ const { catcher } = useToastError()
21
20
const { data : general, isLoading : isLoadingGeneralSettings, silentlyRefresh : refreshGeneral } = useApi (() => getGeneralSettings ())
22
21
const { data : loginSettings, isLoading : isLoadingLoginSettings, silentlyRefresh : refreshLoginSettings } = useApi (() => getLoginSettings ())
23
22
23
+ const { data : mfaTypes, isLoading : isLoadingMFATypes } = useApi (() => getMFAMethods ())
24
24
25
25
const generalData = computed (() => general .value ?? {} as GeneralSettingsDTO )
26
26
const loginSettingsData = computed (() => loginSettings .value ?? {} as LoginSettingsDTO )
27
27
28
+ const textValue = ref (general .value ?.dns .join (' \n ' ) ?? ' ' )
28
29
29
- const { Input : Dns, Arr : DnsArr } = useTextareaInput ()
30
-
31
- Dns .value = (general .value ?.dns ?? []).join (" \n " )
30
+ watch (textValue , (newValue ) => {
31
+ if (general .value ) {
32
+ general .value .dns = newValue .split (' \n ' ).filter (item => item .trim () !== ' ' )
33
+ }
34
+ })
32
35
36
+ watch (general , (newValue ) => {
37
+ if (newValue ) {
38
+ textValue .value = newValue .dns .join (' \n ' )
39
+ }
40
+ })
33
41
34
42
async function saveGeneralSettings() {
35
43
try {
@@ -50,7 +58,7 @@ async function saveGeneralSettings() {
50
58
async function saveLoginSettings() {
51
59
try {
52
60
const resp = await updateLoginSettings (loginSettingsData .value as LoginSettingsDTO )
53
- refreshGeneral ()
61
+ refreshLoginSettings ()
54
62
55
63
if (! resp .success ) {
56
64
toast .error (resp .message ?? ' Failed' )
@@ -68,13 +76,13 @@ async function saveLoginSettings() {
68
76
69
77
<template >
70
78
<main class =" w-full p-4" >
71
- <PageLoading v-if =" isLoadingGeneralSettings || isLoadingLoginSettings" />
79
+ <PageLoading v-if =" isLoadingGeneralSettings || isLoadingLoginSettings || isLoadingMFATypes " />
72
80
73
81
<div v-else >
74
82
<h1 class =" text-4xl font-bold" >Settings</h1 >
75
83
<div class =" mt-6 flex flex-wrap gap-6" >
76
- <div class =" flex w-full gap-4" >
77
- <div class =" card bg-base-100 shadow-xl min-w-[400px ]" >
84
+ <div class =" flex flex-wrap w-full gap-4" >
85
+ <div class =" card bg-base-100 shadow-xl min-w-[350px ]" >
78
86
<div class =" card-body" >
79
87
<h2 class =" card-title" >General</h2 >
80
88
@@ -103,48 +111,155 @@ async function saveLoginSettings() {
103
111
<div class =" form-control" >
104
112
<label for =" dns" class =" block font-medium text-gray-900 pt-6" >DNS</label >
105
113
<textarea class =" rules-input textarea textarea-bordered w-full font-mono" rows =" 3"
106
- v-model =" Dns " ></textarea >
114
+ v-model =" textValue " ></textarea >
107
115
</div >
108
116
117
+ <div class =" flex flex-grow" ></div >
109
118
110
- <button type =" submit" class =" btn btn-primary w-full"
111
- @click =" () => saveGeneralSettings()" >
112
- <span class =" loading loading-spinner loading-md" v-if =" isLoadingGeneralSettings" ></span >
113
- Save
114
- </button >
119
+ <button type =" submit" class =" btn btn-primary w-full" @click =" () => saveGeneralSettings()" >
120
+ <span class =" loading loading-spinner loading-md" v-if =" isLoadingGeneralSettings" ></span >
121
+ Save
122
+ </button >
115
123
</div >
116
124
</div >
117
- <div class =" card bg-base-100 shadow-xl min-w-[400px ]" >
125
+ <div class =" card bg-base-100 shadow-xl min-w-[350px ]" >
118
126
<div class =" card-body" >
119
127
<h2 class =" card-title" >Login</h2 >
120
128
121
129
<div class =" form-control" >
122
130
<label class =" label font-bold" >
123
131
<span class =" label-text" >Session Life Time (Minutes)</span >
124
132
</label >
125
- <input v-model =" loginSettingsData.max_session_lifetime_minutes" type =" number" class =" input input-bordered w-full" />
133
+ <input v-model =" loginSettingsData.max_session_lifetime_minutes" type =" number"
134
+ class =" input input-bordered w-full" />
126
135
</div >
127
136
128
137
<div class =" form-control" >
129
138
<label class =" label font-bold" >
130
139
<span class =" label-text" >Inactivity Timeout (Minutes)</span >
131
140
</label >
132
- <input v-model =" loginSettingsData.session_inactivity_timeout_minutes" type =" number" class =" input input-bordered w-full" />
141
+ <input v-model =" loginSettingsData.session_inactivity_timeout_minutes" type =" number"
142
+ class =" input input-bordered w-full" />
133
143
</div >
134
144
135
145
<div class =" form-control" >
136
146
<label class =" label font-bold" >
137
147
<span class =" label-text" >Max Authentication Attempts</span >
138
148
</label >
139
- <input v-model =" loginSettingsData.lockout" type =" number"
149
+ <input v-model =" loginSettingsData.lockout" type =" number" class =" input input-bordered w-full" />
150
+ </div >
151
+
152
+ <div class =" form-control w-full" >
153
+ <label for =" default_method" class =" label font-bold" >Default MFA Method</label >
154
+ <select class =" select select-bordered " name =" default_method"
155
+ v-model =" loginSettingsData.default_mfa_method" >
156
+ <option v-for =" method in loginSettingsData.enabled_mfa_methods"
157
+ :selected =" method == loginSettingsData.default_mfa_method" :value =" method" >{{ method }}</option >
158
+ </select >
159
+ </div >
160
+
161
+ <div class =" flex flex-col" >
162
+ <div v-for =" method in mfaTypes" class =" form-control w-full" >
163
+ <label :for =method.method class =" label cursor-pointer" >
164
+ <span class =" label-text" >{{ method.friendly_name }}</span >
165
+ <input :name =method.method type =" checkbox" class =" toggle toggle-primary" :value =" method.method"
166
+ v-model =" loginSettingsData.enabled_mfa_methods"
167
+ :checked =" loginSettingsData.enabled_mfa_methods.indexOf(method.method) != -1" />
168
+ </label >
169
+ </div >
170
+ </div >
171
+
172
+ <div class =" flex flex-grow" ></div >
173
+
174
+ <button type =" submit" class =" btn btn-primary w-full" @click =" () => saveLoginSettings()" >
175
+ <span class =" loading loading-spinner loading-md" v-if =" isLoadingLoginSettings" ></span >
176
+ Save
177
+ </button >
178
+ </div >
179
+ </div >
180
+
181
+ <div >
182
+ <div
183
+ v-if =" loginSettingsData.enabled_mfa_methods.indexOf('totp') != -1 || loginSettingsData.enabled_mfa_methods.indexOf('webauthn') != -1"
184
+ class =" card bg-base-100 shadow-xl min-w-[350px] h-max mb-4" >
185
+ <div class =" card-body" >
186
+ <h2 class =" card-title" >Login > General</h2 >
187
+
188
+ <div class =" form-control" >
189
+ <label class =" label font-bold" >
190
+ <span class =" label-text" >Issuer</span >
191
+ </label >
192
+ <input v-model =" loginSettingsData.issuer" type =" text" required class =" input input-bordered w-full" />
193
+ </div >
194
+ <div class =" form-control" >
195
+ <label class =" label font-bold" >
196
+ <span class =" label-text" >Internal VPN Domain</span >
197
+ </label >
198
+ <input v-model =" loginSettingsData.domain" type =" text" required class =" input input-bordered w-full" />
199
+ </div >
200
+ </div >
201
+ </div >
202
+
203
+ <div v-if =" loginSettingsData.enabled_mfa_methods.indexOf('pam') != -1"
204
+ class =" card bg-base-100 shadow-xl min-w-[350px] h-max" >
205
+ <div class =" card-body" >
206
+ <h2 class =" card-title" >Login > System Login</h2 >
207
+
208
+ <div class =" form-control" >
209
+ <label class =" label font-bold" >
210
+ <span class =" label-text" >Service Name</span >
211
+ </label >
212
+ <input v-model =" loginSettingsData.pam.service_name" type =" text" required
213
+ class =" input input-bordered w-full" />
214
+ </div >
215
+ </div >
216
+ </div >
217
+ </div >
218
+
219
+ <div v-if =" loginSettingsData.enabled_mfa_methods.indexOf('oidc') != -1"
220
+ class =" card bg-base-100 shadow-xl min-w-[350px] h-max" >
221
+ <div class =" card-body" >
222
+ <h2 class =" card-title" >Login > SSO Settings</h2 >
223
+
224
+ <div class =" form-control" >
225
+ <label class =" label font-bold" >
226
+ <span class =" label-text" >Provider URL</span >
227
+ </label >
228
+ <input v-model =" loginSettingsData.oidc.issuer" type =" text" required
229
+ class =" input input-bordered w-full" />
230
+ </div >
231
+
232
+ <div class =" form-control" >
233
+ <label class =" label font-bold" >
234
+ <span class =" label-text" >Client ID</span >
235
+ </label >
236
+ <input v-model =" loginSettingsData.oidc.client_id" type =" text" required
237
+ class =" input input-bordered w-full" />
238
+ </div >
239
+
240
+ <div class =" form-control" >
241
+ <label class =" label font-bold" >
242
+ <span class =" label-text" >Client Secret</span >
243
+ </label >
244
+ <input v-model =" loginSettingsData.oidc.client_secret" type =" password" required
140
245
class =" input input-bordered w-full" />
141
246
</div >
142
247
143
- <button type =" submit" class =" btn btn-primary w-full"
144
- @click =" () => saveLoginSettings()" >
145
- <span class =" loading loading-spinner loading-md" v-if =" isLoadingLoginSettings" ></span >
146
- Save
147
- </button >
248
+ <div class =" form-control" >
249
+ <label class =" label font-bold" >
250
+ <span class =" label-text" >Groups Claim Name</span >
251
+ </label >
252
+ <input v-model =" loginSettingsData.oidc.group_claim_name" type =" text"
253
+ class =" input input-bordered w-full" />
254
+ </div >
255
+
256
+ <div class =" form-control" >
257
+ <label class =" label font-bold" >
258
+ <span class =" label-text" >Device Username Claim</span >
259
+ </label >
260
+ <input v-model =" loginSettingsData.oidc.device_username_claim" type =" text"
261
+ class =" input input-bordered w-full" />
262
+ </div >
148
263
</div >
149
264
</div >
150
265
</div >
0 commit comments