|
@@ -3,128 +3,232 @@
|
|
|
v-layout(row, wrap)
|
|
|
v-flex(xs12)
|
|
|
.admin-header
|
|
|
- img(src='/svg/icon-rest-api.svg', alt='API', style='width: 80px;')
|
|
|
+ img.animated.fadeInUp(src='/svg/icon-rest-api.svg', alt='API', style='width: 80px;')
|
|
|
.admin-header-title
|
|
|
- .headline.blue--text.text--darken-2 API Access
|
|
|
- .subtitle-1.grey--text Manage keys to access the API #[v-chip(label, color='primary', small).white--text coming soon]
|
|
|
+ .headline.primary--text.animated.fadeInLeft {{$t('admin:api.title')}}
|
|
|
+ .subtitle-1.grey--text.animated.fadeInLeft {{$t('admin:api.subtitle')}}
|
|
|
v-spacer
|
|
|
- v-btn(outline, color='grey', large, @click='refresh', disabled)
|
|
|
- v-icon refresh
|
|
|
- v-btn(color='green', disabled, depressed, large, @click='globalSwitch')
|
|
|
- v-icon(left) power_settings_new
|
|
|
- | Enable API
|
|
|
- v-btn(color='primary', depressed, large, @click='newKey', disabled)
|
|
|
- v-icon(left) add
|
|
|
- | New API Key
|
|
|
- v-card.mt-3
|
|
|
- v-data-table(
|
|
|
- v-model='selected'
|
|
|
- :items='items',
|
|
|
- :headers='headers',
|
|
|
- :search='search',
|
|
|
- :pagination.sync='pagination',
|
|
|
- :rows-per-page-items='[15]'
|
|
|
- select-all,
|
|
|
- hide-actions,
|
|
|
- disable-initial-sort
|
|
|
- )
|
|
|
- template(slot='headers', slot-scope='props')
|
|
|
- tr
|
|
|
- th(width='50')
|
|
|
- th.text-xs-right(
|
|
|
- width='80'
|
|
|
- :class='[`column sortable`, pagination.descending ? `desc` : `asc`, pagination.sortBy === `id` ? `active` : ``]'
|
|
|
- @click='changeSort(`id`)'
|
|
|
- )
|
|
|
- v-icon(small) arrow_upward
|
|
|
- | ID
|
|
|
- th.text-xs-left(
|
|
|
- v-for='header in props.headers'
|
|
|
- :key='header.text'
|
|
|
- :width='header.width'
|
|
|
- :class='[`column sortable`, pagination.descending ? `desc` : `asc`, header.value === pagination.sortBy ? `active` : ``]'
|
|
|
- @click='changeSort(header.value)'
|
|
|
- )
|
|
|
- | {{ header.text }}
|
|
|
- v-icon(small) arrow_upward
|
|
|
- template(slot='items', slot-scope='props')
|
|
|
- tr(:active='props.selected')
|
|
|
- td
|
|
|
- v-checkbox(hide-details, :input-value='props.selected', color='blue darken-2', @click='props.selected = !props.selected')
|
|
|
- td.text-xs-right {{ props.item.id }}
|
|
|
- td {{ props.item.name }}
|
|
|
- td {{ props.item.key }}
|
|
|
- td {{ props.item.createdOn }}
|
|
|
- td {{ props.item.updatedOn }}
|
|
|
- td: v-btn(icon): v-icon.grey--text.text--darken-1 more_horiz
|
|
|
- template(slot='no-data')
|
|
|
- v-alert.mt-3(icon='info', :value='true', outline, color='info') No API keys have been generated yet.
|
|
|
- .text-xs-center.py-2
|
|
|
- v-pagination(v-model='pagination.page', :length='pages')
|
|
|
+ template(v-if='enabled')
|
|
|
+ status-indicator.mr-3(positive, pulse)
|
|
|
+ .caption.green--text.animated.fadeInLeft {{$t('admin:api.enabled')}}
|
|
|
+ template(v-else)
|
|
|
+ status-indicator.mr-3(negative, pulse)
|
|
|
+ .caption.red--text.animated.fadeInLeft {{$t('admin:api.disabled')}}
|
|
|
+ v-spacer
|
|
|
+ v-btn.mr-3.animated.fadeInDown.wait-p2s(outlined, color='grey', large, @click='refresh')
|
|
|
+ v-icon mdi-refresh
|
|
|
+ v-btn.mr-3.animated.fadeInDown.wait-p1s(:color='enabled ? `red` : `green`', depressed, large, @click='globalSwitch', dark, :loading='isToggleLoading')
|
|
|
+ v-icon(left) mdi-power
|
|
|
+ span(v-if='!enabled') {{$t('admin:api.enableButton')}}
|
|
|
+ span(v-else) {{$t('admin:api.disableButton')}}
|
|
|
+ v-btn.animated.fadeInDown(color='primary', depressed, large, @click='newKey', dark)
|
|
|
+ v-icon(left) mdi-plus
|
|
|
+ span {{$t('admin:api.newKeyButton')}}
|
|
|
+ v-card.mt-3.animated.fadeInUp
|
|
|
+ v-simple-table(v-if='keys && keys.length > 0')
|
|
|
+ template(v-slot:default)
|
|
|
+ thead
|
|
|
+ tr.grey(:class='$vuetify.theme.dark ? `darken-4-d5` : `lighten-5`')
|
|
|
+ th {{$t('admin:api.headerName')}}
|
|
|
+ th {{$t('admin:api.headerKeyEnding')}}
|
|
|
+ th {{$t('admin:api.headerExpiration')}}
|
|
|
+ th {{$t('admin:api.headerCreated')}}
|
|
|
+ th {{$t('admin:api.headerLastUpdated')}}
|
|
|
+ th(width='100') {{$t('admin:api.headerRevoke')}}
|
|
|
+ tbody
|
|
|
+ tr(v-for='key of keys', :key='`key-` + key.id')
|
|
|
+ td
|
|
|
+ strong(:class='key.isRevoked ? `red--text` : ``') {{ key.name }}
|
|
|
+ em.caption.ml-1.red--text(v-if='key.isRevoked') (revoked)
|
|
|
+ td.caption {{ key.keyShort }}
|
|
|
+ td(:style='key.isRevoked ? `text-decoration: line-through;` : ``') {{ key.expiration | moment('LL') }}
|
|
|
+ td {{ key.createdAt | moment('calendar') }}
|
|
|
+ td {{ key.updatedAt | moment('calendar') }}
|
|
|
+ td: v-btn(icon, @click='revoke(key)', :disabled='key.isRevoked'): v-icon(color='error') mdi-cancel
|
|
|
+ v-card-text(v-else)
|
|
|
+ v-alert.mb-0(icon='mdi-information', :value='true', outlined, color='info') {{$t('admin:api.noKeyInfo')}}
|
|
|
+
|
|
|
+ create-api-key(v-model='isCreateDialogShown', @refresh='refresh(false)')
|
|
|
+
|
|
|
+ v-dialog(v-model='isRevokeConfirmDialogShown', max-width='500', persistent)
|
|
|
+ v-card
|
|
|
+ .dialog-header.is-red {{$t('admin:api.revokeConfirm')}}
|
|
|
+ v-card-text.pa-4
|
|
|
+ i18next(tag='span', path='admin:api.revokeConfirmText')
|
|
|
+ strong(place='name') {{ current.name }}
|
|
|
+ v-card-actions
|
|
|
+ v-spacer
|
|
|
+ v-btn(text, @click='isRevokeConfirmDialogShown = false', :disabled='revokeLoading') {{$t('common:actions.cancel')}}
|
|
|
+ v-btn(color='red', dark, @click='revokeConfirm', :loading='revokeLoading') {{$t('admin:api.revoke')}}
|
|
|
</template>
|
|
|
|
|
|
<script>
|
|
|
+import _ from 'lodash'
|
|
|
+import gql from 'graphql-tag'
|
|
|
+import { StatusIndicator } from 'vue-status-indicator'
|
|
|
+
|
|
|
+import CreateApiKey from './admin-api-create.vue'
|
|
|
+
|
|
|
export default {
|
|
|
+ components: {
|
|
|
+ StatusIndicator,
|
|
|
+ CreateApiKey
|
|
|
+ },
|
|
|
data() {
|
|
|
return {
|
|
|
- selected: [],
|
|
|
- pagination: {},
|
|
|
- items: [],
|
|
|
- headers: [
|
|
|
- { text: 'Name', value: 'name' },
|
|
|
- { text: 'Key', value: 'key' },
|
|
|
- { text: 'Created On', value: 'createdOn' },
|
|
|
- { text: 'Updated On', value: 'updatedOn' },
|
|
|
- { text: '', value: 'actions', sortable: false, width: 50 }
|
|
|
- ],
|
|
|
- search: ''
|
|
|
- }
|
|
|
- },
|
|
|
- computed: {
|
|
|
- pages () {
|
|
|
- if (this.pagination.rowsPerPage == null || this.pagination.totalItems == null) {
|
|
|
- return 0
|
|
|
- }
|
|
|
-
|
|
|
- return Math.ceil(this.pagination.totalItems / this.pagination.rowsPerPage)
|
|
|
+ enabled: false,
|
|
|
+ isToggleLoading: false,
|
|
|
+ keys: [],
|
|
|
+ isCreateDialogShown: false,
|
|
|
+ isRevokeConfirmDialogShown: false,
|
|
|
+ revokeLoading: false,
|
|
|
+ current: {}
|
|
|
}
|
|
|
},
|
|
|
methods: {
|
|
|
- changeSort (column) {
|
|
|
- if (this.pagination.sortBy === column) {
|
|
|
- this.pagination.descending = !this.pagination.descending
|
|
|
- } else {
|
|
|
- this.pagination.sortBy = column
|
|
|
- this.pagination.descending = false
|
|
|
+ async refresh (notify = true) {
|
|
|
+ this.$apollo.queries.keys.refetch()
|
|
|
+ if (notify) {
|
|
|
+ this.$store.commit('showNotification', {
|
|
|
+ message: this.$t('admin:api.refreshSuccess'),
|
|
|
+ style: 'success',
|
|
|
+ icon: 'cached'
|
|
|
+ })
|
|
|
}
|
|
|
},
|
|
|
- toggleAll () {
|
|
|
- if (this.selected.length) {
|
|
|
- this.selected = []
|
|
|
- } else {
|
|
|
- this.selected = this.items.slice()
|
|
|
+ async globalSwitch () {
|
|
|
+ this.isToggleLoading = true
|
|
|
+ try {
|
|
|
+ const resp = await this.$apollo.mutate({
|
|
|
+ mutation: gql`
|
|
|
+ mutation ($enabled: Boolean!) {
|
|
|
+ authentication {
|
|
|
+ setApiState (enabled: $enabled) {
|
|
|
+ responseResult {
|
|
|
+ succeeded
|
|
|
+ errorCode
|
|
|
+ slug
|
|
|
+ message
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ `,
|
|
|
+ variables: {
|
|
|
+ enabled: !this.enabled
|
|
|
+ },
|
|
|
+ watchLoading (isLoading) {
|
|
|
+ this.$store.commit(`loading${isLoading ? 'Start' : 'Stop'}`, 'admin-api-toggle')
|
|
|
+ }
|
|
|
+ })
|
|
|
+ if (_.get(resp, 'data.authentication.setApiState.responseResult.succeeded', false)) {
|
|
|
+ this.$store.commit('showNotification', {
|
|
|
+ style: 'success',
|
|
|
+ message: this.enabled ? this.$t('admin:api.toggleStateDisabledSuccess') : this.$t('admin:api.toggleStateEnabledSuccess'),
|
|
|
+ icon: 'check'
|
|
|
+ })
|
|
|
+ await this.$apollo.queries.enabled.refetch()
|
|
|
+ } else {
|
|
|
+ this.$store.commit('showNotification', {
|
|
|
+ style: 'red',
|
|
|
+ message: _.get(resp, 'data.authentication.setApiState.responseResult.message', 'An unexpected error occured.'),
|
|
|
+ icon: 'alert'
|
|
|
+ })
|
|
|
+ }
|
|
|
+ } catch (err) {
|
|
|
+ this.$store.commit('pushGraphError', err)
|
|
|
}
|
|
|
+ this.isToggleLoading = false
|
|
|
},
|
|
|
- async refresh() {
|
|
|
- this.$store.commit('showNotification', {
|
|
|
- style: 'indigo',
|
|
|
- message: `Coming soon...`,
|
|
|
- icon: 'directions_boat'
|
|
|
- })
|
|
|
+ async newKey () {
|
|
|
+ this.isCreateDialogShown = true
|
|
|
},
|
|
|
- async globalSwitch() {
|
|
|
- this.$store.commit('showNotification', {
|
|
|
- style: 'indigo',
|
|
|
- message: `Coming soon...`,
|
|
|
- icon: 'directions_boat'
|
|
|
- })
|
|
|
+ revoke (key) {
|
|
|
+ this.current = key
|
|
|
+ this.isRevokeConfirmDialogShown = true
|
|
|
+ },
|
|
|
+ async revokeConfirm () {
|
|
|
+ this.revokeLoading = true
|
|
|
+ try {
|
|
|
+ const resp = await this.$apollo.mutate({
|
|
|
+ mutation: gql`
|
|
|
+ mutation ($id: Int!) {
|
|
|
+ authentication {
|
|
|
+ revokeApiKey (id: $id) {
|
|
|
+ responseResult {
|
|
|
+ succeeded
|
|
|
+ errorCode
|
|
|
+ slug
|
|
|
+ message
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ `,
|
|
|
+ variables: {
|
|
|
+ id: this.current.id
|
|
|
+ },
|
|
|
+ watchLoading (isLoading) {
|
|
|
+ this.$store.commit(`loading${isLoading ? 'Start' : 'Stop'}`, 'admin-api-revoke')
|
|
|
+ }
|
|
|
+ })
|
|
|
+ if (_.get(resp, 'data.authentication.revokeApiKey.responseResult.succeeded', false)) {
|
|
|
+ this.$store.commit('showNotification', {
|
|
|
+ style: 'success',
|
|
|
+ message: this.$t('admin:api.revokeSuccess'),
|
|
|
+ icon: 'check'
|
|
|
+ })
|
|
|
+ this.refresh(false)
|
|
|
+ } else {
|
|
|
+ this.$store.commit('showNotification', {
|
|
|
+ style: 'red',
|
|
|
+ message: _.get(resp, 'data.authentication.revokeApiKey.responseResult.message', 'An unexpected error occured.'),
|
|
|
+ icon: 'alert'
|
|
|
+ })
|
|
|
+ }
|
|
|
+ } catch (err) {
|
|
|
+ this.$store.commit('pushGraphError', err)
|
|
|
+ }
|
|
|
+ this.isRevokeConfirmDialogShown = false
|
|
|
+ this.revokeLoading = false
|
|
|
+ }
|
|
|
+ },
|
|
|
+ apollo: {
|
|
|
+ enabled: {
|
|
|
+ query: gql`
|
|
|
+ {
|
|
|
+ authentication {
|
|
|
+ apiState
|
|
|
+ }
|
|
|
+ }
|
|
|
+ `,
|
|
|
+ fetchPolicy: 'network-only',
|
|
|
+ update: (data) => data.authentication.apiState,
|
|
|
+ watchLoading (isLoading) {
|
|
|
+ this.$store.commit(`loading${isLoading ? 'Start' : 'Stop'}`, 'admin-api-state-refresh')
|
|
|
+ }
|
|
|
},
|
|
|
- async newKey() {
|
|
|
- this.$store.commit('showNotification', {
|
|
|
- style: 'indigo',
|
|
|
- message: `Coming soon...`,
|
|
|
- icon: 'directions_boat'
|
|
|
- })
|
|
|
+ keys: {
|
|
|
+ query: gql`
|
|
|
+ {
|
|
|
+ authentication {
|
|
|
+ apiKeys {
|
|
|
+ id
|
|
|
+ name
|
|
|
+ keyShort
|
|
|
+ expiration
|
|
|
+ isRevoked
|
|
|
+ createdAt
|
|
|
+ updatedAt
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ `,
|
|
|
+ fetchPolicy: 'network-only',
|
|
|
+ update: (data) => data.authentication.apiKeys,
|
|
|
+ watchLoading (isLoading) {
|
|
|
+ this.$store.commit(`loading${isLoading ? 'Start' : 'Stop'}`, 'admin-api-keys-refresh')
|
|
|
+ }
|
|
|
}
|
|
|
}
|
|
|
}
|