more folders cleanup
This commit is contained in:
119
inspectr/app/viewer/app.js
Normal file
119
inspectr/app/viewer/app.js
Normal file
@@ -0,0 +1,119 @@
|
||||
/* global Vue axios */ //> from vue.html
|
||||
const GET = (url) => axios.get('/odata/v4/-data'+url)
|
||||
const storageGet = (key, def) => localStorage.getItem('data-viewer:'+key) || def
|
||||
const storageSet = (key, val) => localStorage.setItem('data-viewer:'+key, val)
|
||||
const columnKeysFirst = (c1, c2) => {
|
||||
if (c1.isKey && !c2.isKey) return -1
|
||||
if (!c1.isKey && c2.isKey) return 1
|
||||
if (c1.isKey && c2.isKey) return c1.name.localeCompare(c2.name)
|
||||
return 0 // retain natural order of normal columns
|
||||
}
|
||||
|
||||
const vue = Vue.createApp ({
|
||||
|
||||
data() { return {
|
||||
error: undefined,
|
||||
dataSource: storageGet('data-source', 'db'),
|
||||
skip: storageGet('skip', 0),
|
||||
top: storageGet('top', 20),
|
||||
entity: storageGet('entity') ? JSON.parse(storageGet('entity')) : undefined,
|
||||
entities: [],
|
||||
columns: [],
|
||||
data: [],
|
||||
rowDetails: {},
|
||||
rowKey: storageGet('rowKey')
|
||||
}},
|
||||
|
||||
watch: {
|
||||
dataSource: (v) => { storageSet('data-source', v); vue.fetchEntities() },
|
||||
skip: (v) => { storageSet('skip', v); if (vue.entity) vue.fetchData() },
|
||||
top: (v) => { storageSet('top', v); if (vue.entity) vue.fetchData() },
|
||||
},
|
||||
|
||||
methods: {
|
||||
|
||||
async fetchEntities () {
|
||||
let url = `/Entities`
|
||||
if (vue.dataSource === 'db') url += `?dataSource=db`
|
||||
const {data} = await GET(url)
|
||||
vue.entities = data.value
|
||||
vue.entities.forEach(entity => entity.columns.sort(columnKeysFirst))
|
||||
const entity = vue.entity && vue.entities.find(e => e.name === vue.entity.name)
|
||||
if (entity) { // restore selection from previous fetch
|
||||
vue.columns = entity.columns
|
||||
await vue.fetchData(entity)
|
||||
} else {
|
||||
vue.entity = undefined
|
||||
vue.columns = []
|
||||
vue.data = []
|
||||
vue.rowDetails = {}
|
||||
}
|
||||
},
|
||||
|
||||
async inspectEntity (eve) {
|
||||
const entity = vue.entity = vue.entities [eve.currentTarget.rowIndex-1]
|
||||
storageSet('entity', JSON.stringify(entity))
|
||||
vue.columns = vue.entities.find(e => e.name === entity.name).columns
|
||||
return await this.fetchData()
|
||||
},
|
||||
|
||||
async fetchData () {
|
||||
let url = `/Data?entity=${vue.entity.name}&$skip=${vue.skip}&$top=${vue.top}`
|
||||
if (vue.dataSource === 'db') url += `&dataSource=db`
|
||||
|
||||
try {
|
||||
const {data} = await GET(url)
|
||||
// sort data along column order
|
||||
const columnIndexes = {}
|
||||
vue.columns.forEach((col, i) => columnIndexes[col.name] = i)
|
||||
vue.data = data.value.map(d => d.record
|
||||
.sort((r1, r2) => columnIndexes[r1.column] - columnIndexes[r2.column])
|
||||
.map(r => r.data)
|
||||
)
|
||||
const row = vue.data.find(data => vue._makeRowKey(data) === vue.rowKey)
|
||||
if (row) vue._setRowDetails(row)
|
||||
else vue.rowDetails = {}
|
||||
vue.error = undefined
|
||||
} catch (err) {
|
||||
vue.data = []
|
||||
vue.rowDetails = {}
|
||||
if (err.response?.data?.error) {
|
||||
vue.error = err.response.data.error
|
||||
} else {
|
||||
vue.error = { code:err.code, message:err.message }
|
||||
}
|
||||
}
|
||||
|
||||
},
|
||||
|
||||
inspectRow (eve) {
|
||||
vue.rowDetails = {}
|
||||
const selectedRow = eve.currentTarget.rowIndex-1
|
||||
vue.rowKey = vue._makeRowKey(vue.data[selectedRow])
|
||||
storageSet('rowKey', vue.rowKey)
|
||||
vue._setRowDetails(vue.data[selectedRow])
|
||||
},
|
||||
|
||||
_setRowDetails(row) {
|
||||
vue.rowDetails = {}
|
||||
row.forEach((line, colIndex) => {
|
||||
vue.rowDetails[vue.columns[colIndex].name] = line
|
||||
})
|
||||
},
|
||||
|
||||
_makeRowKey(row) {
|
||||
// to identify a row, build a key string out of all key columns' values
|
||||
return row
|
||||
.filter((_, colIndex) => vue.columns[colIndex] && vue.columns[colIndex].isKey)
|
||||
.reduce(((prev, next) => prev += next), '')
|
||||
},
|
||||
|
||||
isActiveRow(row) {
|
||||
return vue._makeRowKey(row) === vue.rowKey
|
||||
}
|
||||
|
||||
}
|
||||
})
|
||||
.mount('#app')
|
||||
|
||||
vue.fetchEntities()
|
||||
95
inspectr/app/viewer/index.html
Normal file
95
inspectr/app/viewer/index.html
Normal file
@@ -0,0 +1,95 @@
|
||||
<!DOCTYPE html>
|
||||
<html>
|
||||
|
||||
<head>
|
||||
<title>Data Browser</title>
|
||||
<link rel="stylesheet" href="https://unpkg.com/primitive-ui/dist/css/main.css">
|
||||
<script src="https://cdn.jsdelivr.net/npm/axios/dist/axios.min.js"></script>
|
||||
<script src="https://cdn.jsdelivr.net/npm/vue@3/dist/vue.global.prod.js"></script>
|
||||
<script src="app.js" defer></script>
|
||||
|
||||
<style>
|
||||
th { position: sticky; top:0; z-index: 2; background-color: white; }
|
||||
.noscroll { overflow: hidden; }
|
||||
.hovering tr:hover td { background: #ebeefc; cursor: pointer; }
|
||||
.highlight { background: #ebeefc !important; }
|
||||
.rating-stars { color:teal }
|
||||
.succeeded { color:teal }
|
||||
.failed { color:red }
|
||||
.condensed { max-width: 100px; text-overflow: ellipsis; white-space: nowrap; }
|
||||
.key { font-weight: bold }
|
||||
.not-key { font-weight: lighter;}
|
||||
.with-sidebar { display: flex; flex-wrap: wrap; gap: 1rem; }
|
||||
.sidebar { flex-basis: 20rem; flex-grow: 1; }
|
||||
.sidebar-main { height: 100vh; overflow-y: scroll; }
|
||||
.not-sidebar { flex-basis: 0; flex-grow: 999; min-inline-size: 50%; align-items: stretch;}
|
||||
.not-sidebar-main { max-height: 40vh; overflow-y: scroll; }
|
||||
.not-sidebar-sub { max-height: 40vh; overflow-y: scroll; }
|
||||
.horizontal label { display: inline; }
|
||||
.horizontal input { width: initial; display: inline; }
|
||||
.error { color: red; }
|
||||
</style>
|
||||
</head>
|
||||
|
||||
<body class="noscroll">
|
||||
<div id='app' class="full-container">
|
||||
|
||||
<h1>Data Browser – {{ entity ? entity.name : '' }}</h1>
|
||||
|
||||
<div class="with-sidebar">
|
||||
<div class="sidebar">
|
||||
<div class="horizontal" style="padding: 0.75rem 0;">
|
||||
<label>Datasource:</label>
|
||||
<input type="radio" id="dataSource-db" value="db" v-model="dataSource">
|
||||
<label for="dataSource-db">Database</label>
|
||||
<input type="radio" id="dataSource-srv" value="service" v-model="dataSource">
|
||||
<label for="dataSource-srv">Service</label>
|
||||
</div>
|
||||
<div class="sidebar-main">
|
||||
<table id='entities' class="hovering">
|
||||
<thead>
|
||||
<th>Entity Name</th>
|
||||
</thead>
|
||||
<tr v-for="e in entities" :key="e.name" @click="inspectEntity" :class="{'highlight': (entity && e.name === entity.name)}">
|
||||
<td>{{ e.name }}</td>
|
||||
</tr>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="not-sidebar">
|
||||
<div class="horizontal">
|
||||
<label for="skip">Skip:</label>
|
||||
<input id="skip" v-model.lazy="skip" title="No. of entries to skip" type="number" min="0">
|
||||
<label for="top">Top:</label>
|
||||
<input id="top" v-model.lazy="top" title="No. of entries to read" type="number" min="0">
|
||||
</div>
|
||||
<div v-if="data" class="not-sidebar-main">
|
||||
<table id='data' class="hovering striped-table condensed">
|
||||
<thead>
|
||||
<th v-for="col in columns" :title="col.type" :class="[col.isKey ? 'key' : 'not-key']">{{ col.name }} </th>
|
||||
</thead>
|
||||
<tr v-for="row in data" @click="inspectRow" :class="{'highlight': isActiveRow(row)}">
|
||||
<td v-for="d in row" :title="d">{{ d }}</td>
|
||||
</tr>
|
||||
</table>
|
||||
</div>
|
||||
<div v-if="error" class="not-sidebar-main error">
|
||||
Error: {{ error.code ? error.code + ' – ' + error.message : error.message }}
|
||||
</div>
|
||||
<p></p>
|
||||
<div v-if="rowDetails" class="not-sidebar-sub">
|
||||
<table id='rowDetails'>
|
||||
<tr v-for="(key, value) in rowDetails" >
|
||||
<td class="key">{{ value }}</td>
|
||||
<td>{{ key }}</td>
|
||||
</tr>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</body>
|
||||
|
||||
</html>
|
||||
4
inspectr/cds-plugin.js
Normal file
4
inspectr/cds-plugin.js
Normal file
@@ -0,0 +1,4 @@
|
||||
const cds = require("@sap/cds")
|
||||
cds.on ('served', ()=> {
|
||||
cds.app.serve ('/data') .from ('@capire/data-viewer','app/viewer')
|
||||
})
|
||||
1
inspectr/index.cds
Normal file
1
inspectr/index.cds
Normal file
@@ -0,0 +1 @@
|
||||
using from './srv/data-service';
|
||||
13
inspectr/package.json
Normal file
13
inspectr/package.json
Normal file
@@ -0,0 +1,13 @@
|
||||
{
|
||||
"name": "@capire/data-viewer",
|
||||
"version": "0.1.0",
|
||||
"description": "A generic browser for data",
|
||||
"dependencies": {
|
||||
"@sap/cds": ">=5.0.4"
|
||||
},
|
||||
"files": [
|
||||
"app",
|
||||
"srv",
|
||||
"index.cds"
|
||||
]
|
||||
}
|
||||
30
inspectr/srv/data-service.cds
Normal file
30
inspectr/srv/data-service.cds
Normal file
@@ -0,0 +1,30 @@
|
||||
/**
|
||||
* Exposes data + entity metadata
|
||||
*/
|
||||
@requires:'authenticated-user'
|
||||
@odata service DataService @( path:'-data' ) {
|
||||
|
||||
/**
|
||||
* Metadata like name and columns/elements
|
||||
*/
|
||||
entity Entities @cds.persistence.skip {
|
||||
key name : String;
|
||||
columns: Composition of many {
|
||||
name : String;
|
||||
type : String;
|
||||
isKey: Boolean;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* The actual data, organized by column name
|
||||
*/
|
||||
entity Data @cds.persistence.skip {
|
||||
key ID : String; // to be OData-compliant
|
||||
record : array of {
|
||||
column : String;
|
||||
data : String;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
63
inspectr/srv/data-service.js
Normal file
63
inspectr/srv/data-service.js
Normal file
@@ -0,0 +1,63 @@
|
||||
const cds = require('@sap/cds')
|
||||
const log = cds.log('data')
|
||||
|
||||
class DataService extends cds.ApplicationService { init(){
|
||||
|
||||
this.on ('READ', 'Entities', req => {
|
||||
const { dataSource } = req.req.query
|
||||
const srvPrefixes = cds.db.model.all('service').map(srv => srv.name+'.')
|
||||
const dataSourceFilter = dataSource === 'db'
|
||||
? e => e['@cds.persistence.skip'] !== true // for DB, excl. entities w/o persistence
|
||||
: e => !!srvPrefixes.find(srvName => e.name.startsWith(srvName)) // only entities reachable from a service
|
||||
|
||||
return cds.db.model.all('entity')
|
||||
.filter (e => req.data && req.data.name ? e.name === req.data.name : true) // honor name filter from request, if any
|
||||
.filter (e => !e.name.startsWith('DRAFT.')) // exclude synthetic stuff
|
||||
.filter (e => !e.name.startsWith('DataService.')) // exclude this service
|
||||
.filter (dataSourceFilter)
|
||||
.sort((e1, e2) => e1.name.localeCompare(e2.name))
|
||||
.map(e => {
|
||||
const columns = Object.entries(e.elements)
|
||||
.filter(([,el]) => !(el instanceof cds.Association)) // exclude assocs+compositions
|
||||
.map(([name, el]) => { return { name, type: el.type, isKey:!!el.key }})
|
||||
return { name: e.name, columns }
|
||||
})
|
||||
})
|
||||
|
||||
this.on ('READ', 'Data', async req => {
|
||||
const { entity: entityName, dataSource: dataSourceName } = req.req.query
|
||||
if (!entityName) return req.reject(400, `Must provide 'entity' query`)
|
||||
const entity = cds.db.model.definitions[entityName]
|
||||
if (!entity) return req.reject(404, 'No such entity: ' + entityName)
|
||||
|
||||
const query = SELECT.from(entity)
|
||||
query.SELECT.limit = req.query.SELECT.limit // forward $skip / $top
|
||||
|
||||
const dataSource = findDataSource(dataSourceName, entityName)
|
||||
let res = await dataSource.run(query)
|
||||
if (!Array.isArray(res)) res = [res] // singleton result
|
||||
return res.map((line) => {
|
||||
const record = Object.entries(line).map(([column, data]) => ({ column, data }))
|
||||
return {
|
||||
record,
|
||||
ID: cds.utils.uuid() // just to be OData-compliant
|
||||
}
|
||||
})
|
||||
})
|
||||
|
||||
return super.init()
|
||||
}}
|
||||
|
||||
module.exports = { DataService }
|
||||
|
||||
/** @returns {cds.Service} */
|
||||
function findDataSource(dataSourceName, entityName) {
|
||||
for (let srv of Object.values(cds.services)) { // all connected services
|
||||
if (!srv.name) continue // FIXME intermediate/pending in cds.services ?
|
||||
if (dataSourceName === srv.name || entityName.startsWith(srv.name+'.')) {
|
||||
log._debug && log.debug(`using ${srv.name} as data source`)
|
||||
return srv
|
||||
}
|
||||
}
|
||||
return cds.services.db // fallback
|
||||
}
|
||||
Reference in New Issue
Block a user