mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-10-27 11:18:14 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			199 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			199 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
<template>
 | 
						|
  <modals-modal ref="modal" v-model="show" name="api-key" :width="800" :height="'unset'" :processing="processing">
 | 
						|
    <template #outer>
 | 
						|
      <div class="absolute top-0 left-0 p-5 w-2/3 overflow-hidden">
 | 
						|
        <p class="text-3xl text-white truncate">{{ title }}</p>
 | 
						|
      </div>
 | 
						|
    </template>
 | 
						|
    <form @submit.prevent="submitForm">
 | 
						|
      <div class="px-4 w-full text-sm py-6 rounded-lg bg-bg shadow-lg border border-black-300 overflow-y-auto overflow-x-hidden" style="min-height: 400px; max-height: 80vh">
 | 
						|
        <div class="w-full p-8">
 | 
						|
          <div class="flex py-2">
 | 
						|
            <div class="w-1/2 px-2">
 | 
						|
              <ui-text-input-with-label v-model.trim="newApiKey.name" :readonly="!isNew" :label="$strings.LabelName" />
 | 
						|
            </div>
 | 
						|
            <div v-if="isNew" class="w-1/2 px-2">
 | 
						|
              <ui-text-input-with-label v-model.trim="newApiKey.expiresIn" :label="$strings.LabelExpiresInSeconds" type="number" :min="0" />
 | 
						|
            </div>
 | 
						|
          </div>
 | 
						|
          <div class="flex items-center pt-4 pb-2 gap-2">
 | 
						|
            <div class="flex items-center px-2">
 | 
						|
              <p class="px-3 font-semibold" id="user-enabled-toggle">{{ $strings.LabelEnable }}</p>
 | 
						|
              <ui-toggle-switch :disabled="isExpired && !apiKey.isActive" labeledBy="user-enabled-toggle" v-model="newApiKey.isActive" />
 | 
						|
            </div>
 | 
						|
            <div v-if="isExpired" class="px-2">
 | 
						|
              <p class="text-sm text-error">{{ $strings.LabelExpired }}</p>
 | 
						|
            </div>
 | 
						|
          </div>
 | 
						|
 | 
						|
          <div class="w-full border-t border-b border-black-200 py-4 px-3 mt-4">
 | 
						|
            <p class="text-lg mb-2 font-semibold">{{ $strings.LabelApiKeyUser }}</p>
 | 
						|
            <p class="text-sm mb-2 text-gray-400">{{ $strings.LabelApiKeyUserDescription }}</p>
 | 
						|
            <ui-select-input v-model="newApiKey.userId" :disabled="isExpired && !apiKey.isActive" :items="userItems" :placeholder="$strings.LabelSelectUser" :label="$strings.LabelApiKeyUser" label-hidden />
 | 
						|
          </div>
 | 
						|
 | 
						|
          <div class="flex pt-4 px-2">
 | 
						|
            <div class="grow" />
 | 
						|
            <ui-btn color="bg-success" type="submit">{{ $strings.ButtonSubmit }}</ui-btn>
 | 
						|
          </div>
 | 
						|
        </div>
 | 
						|
      </div>
 | 
						|
    </form>
 | 
						|
  </modals-modal>
 | 
						|
</template>
 | 
						|
 | 
						|
<script>
 | 
						|
export default {
 | 
						|
  props: {
 | 
						|
    value: Boolean,
 | 
						|
    apiKey: {
 | 
						|
      type: Object,
 | 
						|
      default: () => null
 | 
						|
    },
 | 
						|
    users: {
 | 
						|
      type: Array,
 | 
						|
      default: () => []
 | 
						|
    }
 | 
						|
  },
 | 
						|
  data() {
 | 
						|
    return {
 | 
						|
      processing: false,
 | 
						|
      newApiKey: {},
 | 
						|
      isNew: true
 | 
						|
    }
 | 
						|
  },
 | 
						|
  watch: {
 | 
						|
    show: {
 | 
						|
      handler(newVal) {
 | 
						|
        if (newVal) {
 | 
						|
          this.init()
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
  },
 | 
						|
  computed: {
 | 
						|
    show: {
 | 
						|
      get() {
 | 
						|
        return this.value
 | 
						|
      },
 | 
						|
      set(val) {
 | 
						|
        this.$emit('input', val)
 | 
						|
      }
 | 
						|
    },
 | 
						|
    title() {
 | 
						|
      return this.isNew ? this.$strings.HeaderNewApiKey : this.$strings.HeaderUpdateApiKey
 | 
						|
    },
 | 
						|
    userItems() {
 | 
						|
      return this.users
 | 
						|
        .filter((u) => {
 | 
						|
          // Only show root user if the current user is root
 | 
						|
          return u.type !== 'root' || this.$store.getters['user/getIsRoot']
 | 
						|
        })
 | 
						|
        .map((u) => ({ text: u.username, value: u.id, subtext: u.type }))
 | 
						|
    },
 | 
						|
    isExpired() {
 | 
						|
      if (!this.apiKey || !this.apiKey.expiresAt) return false
 | 
						|
 | 
						|
      return new Date(this.apiKey.expiresAt).getTime() < Date.now()
 | 
						|
    }
 | 
						|
  },
 | 
						|
  methods: {
 | 
						|
    submitForm() {
 | 
						|
      if (!this.newApiKey.name) {
 | 
						|
        this.$toast.error(this.$strings.ToastNameRequired)
 | 
						|
        return
 | 
						|
      }
 | 
						|
 | 
						|
      if (!this.newApiKey.userId) {
 | 
						|
        this.$toast.error(this.$strings.ToastNewApiKeyUserError)
 | 
						|
        return
 | 
						|
      }
 | 
						|
 | 
						|
      if (this.isNew) {
 | 
						|
        this.submitCreateApiKey()
 | 
						|
      } else {
 | 
						|
        this.submitUpdateApiKey()
 | 
						|
      }
 | 
						|
    },
 | 
						|
    submitUpdateApiKey() {
 | 
						|
      if (this.newApiKey.isActive === this.apiKey.isActive && this.newApiKey.userId === this.apiKey.userId) {
 | 
						|
        this.$toast.info(this.$strings.ToastNoUpdatesNecessary)
 | 
						|
        this.show = false
 | 
						|
        return
 | 
						|
      }
 | 
						|
 | 
						|
      const apiKey = {
 | 
						|
        isActive: this.newApiKey.isActive,
 | 
						|
        userId: this.newApiKey.userId
 | 
						|
      }
 | 
						|
 | 
						|
      this.processing = true
 | 
						|
      this.$axios
 | 
						|
        .$patch(`/api/api-keys/${this.apiKey.id}`, apiKey)
 | 
						|
        .then((data) => {
 | 
						|
          this.processing = false
 | 
						|
          if (data.error) {
 | 
						|
            this.$toast.error(`${this.$strings.ToastFailedToUpdate}: ${data.error}`)
 | 
						|
          } else {
 | 
						|
            this.show = false
 | 
						|
            this.$emit('updated', data.apiKey)
 | 
						|
          }
 | 
						|
        })
 | 
						|
        .catch((error) => {
 | 
						|
          this.processing = false
 | 
						|
          console.error('Failed to update apiKey', error)
 | 
						|
          var errMsg = error.response ? error.response.data || '' : ''
 | 
						|
          this.$toast.error(errMsg || this.$strings.ToastFailedToUpdate)
 | 
						|
        })
 | 
						|
    },
 | 
						|
    submitCreateApiKey() {
 | 
						|
      const apiKey = { ...this.newApiKey }
 | 
						|
 | 
						|
      if (this.newApiKey.expiresIn) {
 | 
						|
        apiKey.expiresIn = parseInt(this.newApiKey.expiresIn)
 | 
						|
      } else {
 | 
						|
        delete apiKey.expiresIn
 | 
						|
      }
 | 
						|
 | 
						|
      this.processing = true
 | 
						|
      this.$axios
 | 
						|
        .$post('/api/api-keys', apiKey)
 | 
						|
        .then((data) => {
 | 
						|
          this.processing = false
 | 
						|
          if (data.error) {
 | 
						|
            this.$toast.error(this.$strings.ToastFailedToCreate + ': ' + data.error)
 | 
						|
          } else {
 | 
						|
            this.show = false
 | 
						|
            this.$emit('created', data.apiKey)
 | 
						|
          }
 | 
						|
        })
 | 
						|
        .catch((error) => {
 | 
						|
          this.processing = false
 | 
						|
          console.error('Failed to create apiKey', error)
 | 
						|
          var errMsg = error.response ? error.response.data || '' : ''
 | 
						|
          this.$toast.error(errMsg || this.$strings.ToastFailedToCreate)
 | 
						|
        })
 | 
						|
    },
 | 
						|
    init() {
 | 
						|
      this.isNew = !this.apiKey
 | 
						|
 | 
						|
      if (this.apiKey) {
 | 
						|
        this.newApiKey = {
 | 
						|
          name: this.apiKey.name,
 | 
						|
          isActive: this.apiKey.isActive,
 | 
						|
          userId: this.apiKey.userId
 | 
						|
        }
 | 
						|
      } else {
 | 
						|
        this.newApiKey = {
 | 
						|
          name: null,
 | 
						|
          expiresIn: null,
 | 
						|
          isActive: true,
 | 
						|
          userId: null
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
  },
 | 
						|
  mounted() {}
 | 
						|
}
 | 
						|
</script>
 |