blob: 3f756315c32ed17b37728d4e6002aa69aad1b2b3 [file] [log] [blame]
<!--
Licensed to the Apache Software Foundation (ASF) under one or more
contributor license agreements. See the NOTICE file distributed with
this work for additional information regarding copyright ownership.
The ASF licenses this file to You under the Apache License, Version 2.0
(the "License"); you may not use this file except in compliance with
the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
-->
<template>
<div class="container-fluid">
<div class="row mt-4">
<div class="col-12">
<h2>/{{ datasetName }}</h2>
<div class="card">
<nav class="card-header">
<Menu :dataset-name="datasetName" />
</nav>
<div class="card-body" v-if="services !== null && (!services['gsp-rw'] || services['gsp-rw'].length === 0)">
<div class="alert alert-warning">
No service for adding data available. The Graph Store Protocol service should be configured to allow adding data.
</div>
</div>
<div class="card-body" v-else>
<div v-show="$refs.upload && $refs.upload.dropActive" class="drop-active">
<h3>Drop files to upload</h3>
</div>
<div class="row">
<div class="col-sm-12">
<h3>Upload files {{ postActionUrl }}</h3>
<p>
Load data into the default graph of the currently selected dataset, or the given named graph.
You may upload any RDF format, such as Turtle, RDF/XML or TRiG.
</p>
<form ref="upload-form" novalidate>
<div
id="dataset-graph-name-group"
role="group"
class="form-row form-group"
>
<label
for="dataset-graph-name"
class="col-sm-4 col-md-4 col-lg-2 col-12 col-form-label col-form-label-sm"
>Dataset graph name</label>
<div class="col input-group has-validation">
<input
v-model="datasetGraphName"
:class="graphNameClasses"
id="dataset-graph-name"
ref="dataset-graph-name"
type="text"
placeholder="Leave blank for default graph"
/>
<div class="invalid-feedback">
Invalid graph name. Please remove any spaces.
</div>
</div>
</div>
<div
id="dataset-files"
role="group"
class="form-row form-group"
>
<label
class="col-sm-4 col-md-4 col-lg-2 col-12 col-form-label col-form-label-sm"
>Files to upload</label>
<div class="col has-validation">
<!-- eslint-disable vue/v-on-event-hyphenation -->
<file-upload
ref="upload"
:value="upload.files"
@update:modelValue="upload.files = $event"
:post-action="postActionUrl"
:extensions="upload.extensions"
:accept="upload.accept"
:multiple="upload.multiple"
:directory="upload.directory"
:size="upload.size || 0"
:thread="upload.thread < 1 ? 1 : (upload.thread > 5 ? 5 : upload.thread)"
:headers="upload.headers"
:data="upload.data"
:drop="upload.drop"
:drop-directory="upload.dropDirectory"
:add-index="upload.addIndex"
:class="fileUploadClasses"
:custom-action="handleUploadWithErrorHandling"
>
<FontAwesomeIcon icon="plus" />
<span class="ms-2">select files</span>
</file-upload>
<button
v-if="!$refs.upload || !$refs.upload.active"
@click.prevent="uploadAll()"
type="button"
class="btn btn-primary ms-2 d-inline upload-files"
>
<FontAwesomeIcon icon="upload" />
<span class="ms-2">upload all</span>
</button>
<button
v-else
@click.prevent="$refs.upload.active = false"
type="button"
class="btn btn-primary ms-2 d-inline"
>
<FontAwesomeIcon icon="times-circle" />
<span class="ms-2">stop upload</span>
</button>
<div class="invalid-feedback">
Invalid upload files. Please select at least one file to upload.
</div>
</div>
</div>
<div class="pt-2 pb-2">
<div class="progress" style="height: 1.5rem;">
<div
class="progress-bar"
role="progressbar"
:style="`width: ${uploadSucceededPercentage}%`"
:aria-valuenow="uploadSucceededPercentage"
aria-valuemin="0"
aria-valuemax="100"
>
{{ uploadSucceededCount }}/{{ uploadCount }}
</div>
<div
class="progress-bar bg-danger"
role="progressbar"
:style="`width: ${uploadFailedPercentage}%`"
:aria-valuenow="uploadFailedPercentage"
aria-valuemin="0"
aria-valuemax="100"
>
{{ uploadFailedCount }}/{{ uploadCount }}
</div>
</div>
</div>
</form>
</div>
</div>
<div class="row">
<div class="col">
<jena-table
:fields="datasetTableFields"
:items="datasetTableItems"
empty-text="No files selected"
bordered
fixed
hover
>
<template #cell(size)="data">
{{ readableFileSize(data.item.size) }}
</template>
<template #cell(speed)="data">
{{ readableFileSize(data.item.speed) }}/s
</template>
<template #cell(status)="data">
<div class="progress">
<div
:class="`progress-bar bg-${getFileStatus(data.item)}`"
:style="`width: ${data.item.progress}%`"
:aria-valuenow="`${data.item.progress}`"
aria-valuemin="0"
aria-valuemax="100"
role="progressbar"
>
{{ data.item.progress }}
</div>
</div>
<span class="small">Triples uploaded:&nbsp;</span>
<span v-if="data.item.response.tripleCount" class="small">
{{ data.item.response.tripleCount }}
</span>
<span v-else class="small">0</span>
</template>
<template #cell(actions)="data">
<button
@click.prevent="data.item.success || data.item.error === 'compressing' ? false : $refs.upload.update(data.item, {active: true})"
type="button"
class="btn btn-outline-primary me-0 mb-2 d-block upload-file"
>
<FontAwesomeIcon icon="upload" />
<span class="ms-2">upload now</span>
</button>
<button
@click.prevent="remove(data.item)"
type="button"
class="btn btn-outline-primary me-0 mb-md-0 d-block d-md-inline-block remove-file"
>
<FontAwesomeIcon icon="minus-circle" />
<span class="ms-2">remove</span>
</button>
</template>
</jena-table>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</template>
<script>
import Menu from '@/components/dataset/Menu.vue'
import FileUpload from 'vue-upload-component'
import JenaTable from '@/components/dataset/JenaTable.vue'
import { library } from '@fortawesome/fontawesome-svg-core'
import { faPlus, faUpload, faTimesCircle, faMinusCircle } from '@fortawesome/free-solid-svg-icons'
import { FontAwesomeIcon } from '@fortawesome/vue-fontawesome'
import currentDatasetMixin from '@/mixins/current-dataset'
import currentDatasetMixinNavigationGuards from '@/mixins/current-dataset-navigation-guards'
import { displayError } from '@/utils'
library.add(faPlus, faUpload, faTimesCircle, faMinusCircle)
export default {
name: 'DatasetUpload',
components: {
Menu,
FontAwesomeIcon,
FileUpload,
JenaTable
},
mixins: [
currentDatasetMixin
],
...currentDatasetMixinNavigationGuards,
data () {
return {
datasetGraphName: null,
datasetFiles: null,
graphNameClasses: [
'form-control'
],
fileUploadClasses: [
'btn',
'btn-success'
],
upload: {
files: [],
accept: '', // e.g. 'ttl,xml,rdf,...'
minSize: 0,
// size: 1024 * 1024 * 10,
multiple: true,
directory: null,
drop: true,
dropDirectory: true,
addIndex: false,
thread: 3,
name: 'file',
headers: { // e.g. CSRF headers
},
data: {},
autoCompress: 1024 * 1024,
uploadAuto: false,
isOption: false
// addData: {
// show: false,
// name: '',
// type: '',
// content: ''
// },
// editFile: {
// show: false,
// name: ''
// }
},
datasetTableFields: Object.freeze([
{
key: 'name',
label: 'name',
sortable: true,
sortDirection: 'asc'
},
{
key: 'size',
label: 'size',
sortable: true,
sortDirection: 'asc'
},
{
key: 'speed',
label: 'speed'
},
{
key: 'status',
label: 'status'
},
{
key: 'actions',
label: 'actions'
}
])
}
},
computed: {
datasetTableItems () {
if (!this.upload.files) {
return []
}
return this.upload.files
.map(file => {
return Object.assign(file, {
status: this.getFileStatus(file)
})
})
},
postActionUrl () {
if (this.services === null || !this.services['gsp-rw'] || this.services['gsp-rw'].length === 0) {
return ''
}
const params = (this.datasetGraphName && this.datasetGraphName !== '') ? `?graph=${this.datasetGraphName}` : ''
const dataEndpoint = this.services['gsp-rw']['srv.endpoints'].find(endpoint => endpoint !== '') || ''
return this.$fusekiService.getFusekiUrl(`/${this.datasetName}/${dataEndpoint}${params}`)
},
uploadCount () {
if (!this.upload || !this.upload.files) {
return 0
}
return this.upload.files.length
},
uploadSucceededCount () {
if (!this.upload || !this.upload.files) {
return 0
}
return this.upload.files.filter(f => Boolean(f.success)).length
},
uploadFailedCount () {
if (!this.upload || !this.upload.files) {
return 0
}
return this.upload.files.filter(f => Boolean(f.error)).length
},
uploadFailedPercentage () {
if (this.uploadCount === 0) {
return 0
}
return (this.uploadFailedCount / this.uploadCount) * 100
},
uploadSucceededPercentage () {
if (this.uploadCount === 0) {
return 0
}
return (this.uploadSucceededCount / this.uploadCount) * 100
}
},
watch: {
datasetGraphName () {
this.validateGraphName()
},
upload: {
handler () {
this.validateFiles()
},
deep: true,
immediate: false
}
},
methods: {
getFileStatus (file) {
if (file.error) {
return 'danger'
}
if (file.success) {
return 'success'
}
if (file.active) {
return 'warning'
}
return ''
},
remove (file) {
this.$refs.upload.remove(file)
},
/**
* Return file size in bytes in a human-readable form.
*
* Copied from the Backbone.js code with modifications.
*
* Replaced sprintf usage by .toFixed(2).
*/
readableFileSize: function (size) {
const k = 1024
const m = k * k
if (size >= m) {
return `${(size / m).toFixed(2).replace(/\.?0*$/, '')}mb`
} else if (size >= k) {
return `${(size / k).toFixed(2).replace(/\.?0*$/, '')}kb`
} else {
return `${size} bytes`
}
},
uploadAll () {
if (this.validateForm()) {
this.$refs.upload.active = true
}
},
validateForm () {
return this.validateGraphName() && this.validateFiles()
},
validateGraphName () {
// No spaces allowed in graph names.
const pattern = /^[^\s]+$/
const graphName = this.$refs['dataset-graph-name'].value
if (graphName === '' || pattern.test(graphName)) {
this.graphNameClasses = ['form-control is-valid']
return true
}
this.graphNameClasses = ['form-control is-invalid']
return false
},
validateFiles () {
if (this.upload.files !== null && this.upload.files.length > 0) {
this.fileUploadClasses = [
'btn',
'btn-success',
'is-valid'
]
return true
}
this.fileUploadClasses = [
'btn',
'btn-success',
'is-invalid'
]
return false
},
async handleUploadWithErrorHandling (file, component) {
try {
return component
.uploadHtml5(file)
.catch(error => displayError(this, error))
} catch (error) {
displayError(this, error)
}
}
}
}
</script>