From ddce31eb041f050c5aef906f7c6e9b18ff28b213 Mon Sep 17 00:00:00 2001 From: Martin Weise <martin.weise@tuwien.ac.at> Date: Thu, 26 Dec 2024 06:40:01 +0000 Subject: [PATCH] Dev --- .docs/concepts/authentication.md | 6 +- .docs/concepts/data-versioning.md | 39 +++ .docs/concepts/data-visibility.md | 33 ++ .docs/concepts/messaging.md | 8 +- .docs/concepts/pid.md | 46 +-- .docs/images/architecture.drawio | 47 ++- .docs/images/private-embargo.svg | 3 + .docs/{redirect.html => index.html.tpl} | 8 +- .docs/installation.md | 6 +- .docs/kubernetes.md | 2 +- .gitlab-ci.yml | 5 +- .gitlab/gen-badge.sh | 4 +- README.md | 2 +- build-docs.sh | 2 +- .../dashboards/system.json | 287 +++++++++--------- .../at/tuwien/endpoints/SubsetEndpoint.java | 13 +- .../at/tuwien/endpoints/TableEndpoint.java | 20 +- .../at/tuwien/endpoints/ViewEndpoint.java | 26 +- .../src/main/resources/application.yml | 1 + .../java/at/tuwien/config/CacheConfig.java | 51 +++- .../java/at/tuwien/config/MetricsConfig.java | 24 +- .../gateway/impl/KeycloakGatewayImpl.java | 17 +- .../java/at/tuwien/mapper/MetadataMapper.java | 18 ++ .../at/tuwien/service/MetricsService.java | 10 + .../impl/MetricsServicePrometheusImpl.java | 47 +++ .../impl/QueueServiceRabbitMqImpl.java | 7 +- .../java/at/tuwien/mapper/MetadataMapper.java | 5 +- .../at/tuwien/endpoints/TableEndpoint.java | 3 +- .../tuwien/service/impl/TableServiceImpl.java | 2 +- dbrepo-ui/components/subset/SubsetList.vue | 46 ++- dbrepo-ui/components/view/ViewList.vue | 3 - dbrepo-ui/components/view/ViewToolbar.vue | 2 +- .../[database_id]/view/[view_id]/data.vue | 2 +- helm/dbrepo/README.md | 12 +- lib/python/README.md | 12 +- mkdocs.yml | 8 +- versions.json | 5 + 37 files changed, 526 insertions(+), 306 deletions(-) create mode 100644 .docs/concepts/data-versioning.md create mode 100644 .docs/concepts/data-visibility.md create mode 100644 .docs/images/private-embargo.svg rename .docs/{redirect.html => index.html.tpl} (58%) create mode 100644 dbrepo-data-service/services/src/main/java/at/tuwien/service/MetricsService.java create mode 100644 dbrepo-data-service/services/src/main/java/at/tuwien/service/impl/MetricsServicePrometheusImpl.java diff --git a/.docs/concepts/authentication.md b/.docs/concepts/authentication.md index c49abdf86e..c085e54e15 100644 --- a/.docs/concepts/authentication.md +++ b/.docs/concepts/authentication.md @@ -27,7 +27,7 @@ The [User Interface](../../api/ui) for example refreshes the token on-the-fly by of an expired access token, requests a new one without having to terminate the request. This happens only once after the access token has expired (after e.g. 15 minutes). -## OpenID Connect (OIDC) +## OpenID Connect -We use the widely accepted authentication protocol OIDC for client authentication. Other protocols are e.g. SAML2 which -are not used by DBRepo. \ No newline at end of file +We use the widely accepted authentication protocol OpenID Connect for client authentication. Other protocols are, e.g. +SAML2 which are not used by default in DBRepo. \ No newline at end of file diff --git a/.docs/concepts/data-versioning.md b/.docs/concepts/data-versioning.md new file mode 100644 index 0000000000..8a9abc667b --- /dev/null +++ b/.docs/concepts/data-versioning.md @@ -0,0 +1,39 @@ +--- +author: Martin Weise +--- + +Data is getting bigger and so are expectations of data provisioning in regards to data availability (i.e. immediately +after quality check and not in snapshot intervals), cost-effectiveness (i.e. no duplication of data), transparent, +precise citation and many more. + +[System-versioned](https://mariadb.com/kb/en/system-versioned-tables/) tables in MariaDB are improved data structures +that keep track of historical data. For each entry in a system-versioned table, a time period is maintained that denotes +the validity time span of this tuple from its start to end. Tuples in system-versioned tables are not *actually* +modified, they are marked as (in-)valid in time periods. + +<figure markdown> + +| ID | Sensor | Temp | Start | End | +|----|--------|------|-------|-----| +| 1 | A | 23.1 | t1 | | +| 2 | B | 25.8 | t2 | | + +</figure> + +Assuming that Sensor A was calibrated wrong and an updated measurement is passed to the system-versioned table, the +table contents show that the old row with Temp 23.1 is not deleted, but marked as valid in time span (t1, t3). The +updated row with Temp 22.1 is marked as valid from time span t3 onwards. + +<figure markdown> + +| ID | Sensor | Temp | Start | End | +|----|--------|------|-------|-----| +| 1 | A | 23.1 | t1 | t3 | +| 2 | B | 25.8 | t2 | | +| 1 | A | 22.1 | t3 | | + +</figure> + +System-versioned tables are part of the SQL:2011 standard and have been adopted by many database management system +vendors: MariaDB (10.5 and higher), Google BigQuery, IBM DB2 (12 and higher), SQL Server (2016 and higher), Azure SQL, +PostgreSQL with [temporal tables extension](https://github.com/nearform/temporal_tables), etc. \ No newline at end of file diff --git a/.docs/concepts/data-visibility.md b/.docs/concepts/data-visibility.md new file mode 100644 index 0000000000..e76448ec3e --- /dev/null +++ b/.docs/concepts/data-visibility.md @@ -0,0 +1,33 @@ +--- +author: Martin Weise +--- + +There are several ways to set the visibility of (meta-)data in DBRepo. It is possible to set the data to public/private +and the schema to be public/private for each database and separately for each table, each view and each subset of a +database. + +In total there are three possible scenarios: + +## Public + +!!! info "Possible use-case: data publication supplement to an open-access publication" + +Where the database's data and metadata is set to be *public*. This means everything in the database (tables, views, +subsets) are visible by anyone from the public. + +## Mixed + +!!! info "Possible use-case: private sensor measurements with timed embargo" + +Where the database's data and metadata is set to be *private*. This means everything in the database (tables, views, +subsets) are by default not visible by anyone from the public. You can however make specific views that join tables +and/or filter certain columns and apply a 14-day delay-embargo. + +<figure markdown> + +<figcaption>Figure 1: Public view that joins two private tables and applies a time-embargo</figcaption> +</figure> + +## Private + +!!! info "Possible use-case: data storage for trusted-/virtual research environments" \ No newline at end of file diff --git a/.docs/concepts/messaging.md b/.docs/concepts/messaging.md index c19e1fc2a5..819b20d8a0 100644 --- a/.docs/concepts/messaging.md +++ b/.docs/concepts/messaging.md @@ -21,4 +21,10 @@ JSON, a tuple looks like this in RabbitMQ: ## AMQP DBRepo uses AMQP to route messages which allows for both Basic/Bearer authentication. For more information please -consult the [RabbitMQ AMQP](https://www.rabbitmq.com/tutorials/amqp-concepts) documentation. \ No newline at end of file +consult the [RabbitMQ AMQP](https://www.rabbitmq.com/tutorials/amqp-concepts) documentation. + +## MQTT + +:octicons-tag-16:{ title="Minimum version" } 1.5.0 + +DBRepo supports MQTT for IoT with Basic authentication. \ No newline at end of file diff --git a/.docs/concepts/pid.md b/.docs/concepts/pid.md index e58ed03ac4..892bd65dea 100644 --- a/.docs/concepts/pid.md +++ b/.docs/concepts/pid.md @@ -2,46 +2,6 @@ author: Martin Weise --- -## Data Versioning - -Data is getting bigger and so are expectations of data provisioning in regards to data availability (i.e. immediately -after quality check and not in snapshot intervals), cost-effectiveness (i.e. no duplication of data), transparent, -precise citation and many more. - -[System-versioned](https://mariadb.com/kb/en/system-versioned-tables/) tables in MariaDB are improved data structures -that keep track of historical data. For each entry in a system-versioned table, a time period is maintained that denotes -the validity time span of this tuple from its start to end. Tuples in system-versioned tables are not *actually* -modified, they are marked as (in-)valid in time periods. - -<figure markdown> - -| ID | Sensor | Temp | Start | End | -|----|--------|------|-------|-----| -| 1 | A | 23.1 | t1 | | -| 2 | B | 25.8 | t2 | | - -</figure> - -Assuming that Sensor A was calibrated wrong and an updated measurement is passed to the system-versioned table, the -table contents show that the old row with Temp 23.1 is not deleted, but marked as valid in time span (t1, t3). The -updated row with Temp 22.1 is marked as valid from time span t3 onwards. - -<figure markdown> - -| ID | Sensor | Temp | Start | End | -|----|--------|------|-------|-----| -| 1 | A | 23.1 | t1 | t3 | -| 2 | B | 25.8 | t2 | | -| 1 | A | 22.1 | t3 | | - -</figure> - -System-versioned tables are part of the SQL:2011 standard and have been adopted by many database management system -vendors: MariaDB (10.5 and higher), Google BigQuery, IBM DB2 (12 and higher), SQL Server (2016 and higher), Azure SQL, -PostgreSQL with [temporal tables extension](https://github.com/nearform/temporal_tables), etc. - -## Persistent Identifier - Data in DBRepo always has attached metadata (stored in the [Metadata Database](../../api/metadata-db)). This metadata is provided as machine-understandable context in various open-source formats that is available, even when the original data is not available anymore due to e.g. a retracted dataset (hence the name **persistent**). A persistent identifier @@ -54,4 +14,8 @@ globally, uniquely identifies a data record such as: Combining [data versioning](#data-versioning) and queries, subsets can be precisely identified by storing the query that creates them and the time when the query was executed. We store both in a table inside the database we call the -query store. \ No newline at end of file +query store. + +## DOI + +DBRepo uses the DOI system minted by DataCite, this is optional and not required to function for e.g. test deployments. \ No newline at end of file diff --git a/.docs/images/architecture.drawio b/.docs/images/architecture.drawio index 30bff2a6d0..71536a3212 100644 --- a/.docs/images/architecture.drawio +++ b/.docs/images/architecture.drawio @@ -1,4 +1,4 @@ -<mxfile host="app.diagrams.net" agent="Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/130.0.0.0 Safari/537.36" version="25.0.1" pages="8"> +<mxfile host="Electron" agent="Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) draw.io/25.0.2 Chrome/128.0.6613.186 Electron/32.2.5 Safari/537.36" version="25.0.2" pages="9"> <diagram id="mvBsv1rP8O80Qe3yGnn_" name="docker-compose"> <mxGraphModel dx="683" dy="391" grid="1" gridSize="10" guides="1" tooltips="1" connect="1" arrows="1" fold="1" page="1" pageScale="1" pageWidth="1169" pageHeight="827" math="0" shadow="0"> <root> @@ -1108,4 +1108,49 @@ </root> </mxGraphModel> </diagram> + <diagram id="7HywRA3nQAgvNxZjCRq2" name="private-embargo"> + <mxGraphModel dx="985" dy="394" grid="1" gridSize="10" guides="1" tooltips="1" connect="1" arrows="1" fold="1" page="1" pageScale="1" pageWidth="1169" pageHeight="827" math="0" shadow="0"> + <root> + <mxCell id="0" /> + <mxCell id="1" parent="0" /> + <mxCell id="n6nk3BLY6128t3IB6Ma7-8" style="edgeStyle=orthogonalEdgeStyle;rounded=0;orthogonalLoop=1;jettySize=auto;html=1;entryX=0.25;entryY=0;entryDx=0;entryDy=0;curved=1;exitX=0.5;exitY=1;exitDx=0;exitDy=0;" edge="1" parent="1" source="n6nk3BLY6128t3IB6Ma7-1" target="n6nk3BLY6128t3IB6Ma7-5"> + <mxGeometry relative="1" as="geometry" /> + </mxCell> + <mxCell id="n6nk3BLY6128t3IB6Ma7-11" value="<span style="text-wrap: wrap; background-color: rgb(251, 251, 251);">value,loc_id</span>" style="edgeLabel;html=1;align=center;verticalAlign=middle;resizable=0;points=[];fontSize=8;" vertex="1" connectable="0" parent="n6nk3BLY6128t3IB6Ma7-8"> + <mxGeometry x="0.0303" relative="1" as="geometry"> + <mxPoint as="offset" /> + </mxGeometry> + </mxCell> + <mxCell id="n6nk3BLY6128t3IB6Ma7-1" value="" style="shape=internalStorage;whiteSpace=wrap;html=1;backgroundOutline=1;" vertex="1" parent="1"> + <mxGeometry x="250" y="170" width="80" height="80" as="geometry" /> + </mxCell> + <mxCell id="n6nk3BLY6128t3IB6Ma7-2" value="<b>table</b>: sensor (private)" style="text;html=1;align=center;verticalAlign=middle;whiteSpace=wrap;rounded=0;" vertex="1" parent="1"> + <mxGeometry x="227.5" y="150" width="125" height="20" as="geometry" /> + </mxCell> + <mxCell id="n6nk3BLY6128t3IB6Ma7-9" style="edgeStyle=orthogonalEdgeStyle;rounded=0;orthogonalLoop=1;jettySize=auto;html=1;exitX=0.5;exitY=1;exitDx=0;exitDy=0;entryX=0.75;entryY=0;entryDx=0;entryDy=0;curved=1;" edge="1" parent="1" source="n6nk3BLY6128t3IB6Ma7-3" target="n6nk3BLY6128t3IB6Ma7-5"> + <mxGeometry relative="1" as="geometry" /> + </mxCell> + <mxCell id="n6nk3BLY6128t3IB6Ma7-12" value="id,name,lat,lng" style="edgeLabel;html=1;align=center;verticalAlign=middle;resizable=0;points=[];fontSize=8;" vertex="1" connectable="0" parent="n6nk3BLY6128t3IB6Ma7-9"> + <mxGeometry x="0.1455" relative="1" as="geometry"> + <mxPoint as="offset" /> + </mxGeometry> + </mxCell> + <mxCell id="n6nk3BLY6128t3IB6Ma7-3" value="" style="shape=internalStorage;whiteSpace=wrap;html=1;backgroundOutline=1;" vertex="1" parent="1"> + <mxGeometry x="430" y="170" width="80" height="80" as="geometry" /> + </mxCell> + <mxCell id="n6nk3BLY6128t3IB6Ma7-4" value="<b>table</b>: location (private)" style="text;html=1;align=center;verticalAlign=middle;whiteSpace=wrap;rounded=0;" vertex="1" parent="1"> + <mxGeometry x="405" y="150" width="130" height="20" as="geometry" /> + </mxCell> + <mxCell id="n6nk3BLY6128t3IB6Ma7-5" value="" style="shape=internalStorage;whiteSpace=wrap;html=1;backgroundOutline=1;fontSize=8;" vertex="1" parent="1"> + <mxGeometry x="340" y="290" width="80" height="80" as="geometry" /> + </mxCell> + <mxCell id="n6nk3BLY6128t3IB6Ma7-6" value="<b>view</b>: validated_sensor (public)" style="text;html=1;align=center;verticalAlign=middle;whiteSpace=wrap;rounded=0;" vertex="1" parent="1"> + <mxGeometry x="290" y="370" width="180" height="20" as="geometry" /> + </mxCell> + <mxCell id="n6nk3BLY6128t3IB6Ma7-7" style="edgeStyle=orthogonalEdgeStyle;rounded=0;orthogonalLoop=1;jettySize=auto;html=1;exitX=0.5;exitY=1;exitDx=0;exitDy=0;" edge="1" parent="1" source="n6nk3BLY6128t3IB6Ma7-4" target="n6nk3BLY6128t3IB6Ma7-4"> + <mxGeometry relative="1" as="geometry" /> + </mxCell> + </root> + </mxGraphModel> + </diagram> </mxfile> diff --git a/.docs/images/private-embargo.svg b/.docs/images/private-embargo.svg new file mode 100644 index 0000000000..2c83363264 --- /dev/null +++ b/.docs/images/private-embargo.svg @@ -0,0 +1,3 @@ +<?xml version="1.0" encoding="UTF-8"?> +<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> +<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1" width="308px" height="240px" viewBox="-0.5 -0.5 308 240" style="background-color: rgb(255, 255, 255);"><defs/><rect fill="#ffffff" width="100%" height="100%" x="0" y="0"/><g><g data-cell-id="0"><g data-cell-id="1"><g data-cell-id="n6nk3BLY6128t3IB6Ma7-8"><g><path d="M 62 100 Q 62 120 97 120 Q 132 120 132 133.63" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="stroke"/><path d="M 132 138.88 L 128.5 131.88 L 132 133.63 L 135.5 131.88 Z" fill="rgb(0, 0, 0)" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/></g><g data-cell-id="n6nk3BLY6128t3IB6Ma7-11"><g><g transform="translate(-0.5 -0.5)"><switch><foreignObject pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility" style="overflow: visible; text-align: left;"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 1px; height: 1px; padding-top: 120px; margin-left: 99px;"><div data-drawio-colors="color: rgb(0, 0, 0); background-color: rgb(255, 255, 255); " style="box-sizing: border-box; font-size: 0px; text-align: center;"><div style="display: inline-block; font-size: 8px; font-family: Helvetica; color: rgb(0, 0, 0); line-height: 1.2; pointer-events: all; background-color: rgb(255, 255, 255); white-space: nowrap;"><span style="text-wrap: wrap; background-color: rgb(251, 251, 251);">value,loc_id</span></div></div></div></foreignObject><image x="78" y="115.5" width="42" height="12" xlink:href=""/></switch></g></g></g></g><g data-cell-id="n6nk3BLY6128t3IB6Ma7-1"><g><rect x="22" y="20" width="80" height="80" fill="rgb(255, 255, 255)" stroke="rgb(0, 0, 0)" pointer-events="all"/><path d="M 22 40 L 102 40" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/><path d="M 42 20 L 42 100" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/></g></g><g data-cell-id="n6nk3BLY6128t3IB6Ma7-2"><g><rect x="-0.5" y="0" width="125" height="20" fill="none" stroke="none" pointer-events="all"/></g><g><g transform="translate(-0.5 -0.5)"><switch><foreignObject pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility" style="overflow: visible; text-align: left;"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 123px; height: 1px; padding-top: 10px; margin-left: 1px;"><div data-drawio-colors="color: rgb(0, 0, 0); " style="box-sizing: border-box; font-size: 0px; text-align: center;"><div style="display: inline-block; font-size: 12px; font-family: Helvetica; color: rgb(0, 0, 0); line-height: 1.2; pointer-events: all; white-space: normal; overflow-wrap: normal;"><b>table</b>: sensor (private)</div></div></div></foreignObject><image x="1" y="3.5" width="123" height="17" xlink:href=""/></switch></g></g></g><g data-cell-id="n6nk3BLY6128t3IB6Ma7-9"><g><path d="M 242 100 Q 242 120 207 120 Q 172 120 172 133.63" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="stroke"/><path d="M 172 138.88 L 168.5 131.88 L 172 133.63 L 175.5 131.88 Z" fill="rgb(0, 0, 0)" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/></g><g data-cell-id="n6nk3BLY6128t3IB6Ma7-12"><g><g transform="translate(-0.5 -0.5)"><switch><foreignObject pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility" style="overflow: visible; text-align: left;"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 1px; height: 1px; padding-top: 120px; margin-left: 199px;"><div data-drawio-colors="color: rgb(0, 0, 0); background-color: rgb(255, 255, 255); " style="box-sizing: border-box; font-size: 0px; text-align: center;"><div style="display: inline-block; font-size: 8px; font-family: Helvetica; color: rgb(0, 0, 0); line-height: 1.2; pointer-events: all; background-color: rgb(255, 255, 255); white-space: nowrap;">id,name,lat,lng</div></div></div></foreignObject><image x="173" y="115.5" width="52" height="12" xlink:href=""/></switch></g></g></g></g><g data-cell-id="n6nk3BLY6128t3IB6Ma7-3"><g><rect x="202" y="20" width="80" height="80" fill="rgb(255, 255, 255)" stroke="rgb(0, 0, 0)" pointer-events="all"/><path d="M 202 40 L 282 40" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/><path d="M 222 20 L 222 100" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/></g></g><g data-cell-id="n6nk3BLY6128t3IB6Ma7-4"><g><rect x="177" y="0" width="130" height="20" fill="none" stroke="none" pointer-events="all"/></g><g><g transform="translate(-0.5 -0.5)"><switch><foreignObject pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility" style="overflow: visible; text-align: left;"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 128px; height: 1px; padding-top: 10px; margin-left: 178px;"><div data-drawio-colors="color: rgb(0, 0, 0); " style="box-sizing: border-box; font-size: 0px; text-align: center;"><div style="display: inline-block; font-size: 12px; font-family: Helvetica; color: rgb(0, 0, 0); line-height: 1.2; pointer-events: all; white-space: normal; overflow-wrap: normal;"><b>table</b>: location (private)</div></div></div></foreignObject><image x="178" y="3.5" width="128" height="17" xlink:href=""/></switch></g></g></g><g data-cell-id="n6nk3BLY6128t3IB6Ma7-5"><g><rect x="112" y="140" width="80" height="80" fill="rgb(255, 255, 255)" stroke="rgb(0, 0, 0)" pointer-events="all"/><path d="M 112 160 L 192 160" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/><path d="M 132 140 L 132 220" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/></g></g><g data-cell-id="n6nk3BLY6128t3IB6Ma7-6"><g><rect x="62" y="220" width="180" height="20" fill="none" stroke="none" pointer-events="all"/></g><g><g transform="translate(-0.5 -0.5)"><switch><foreignObject pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility" style="overflow: visible; text-align: left;"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 178px; height: 1px; padding-top: 230px; margin-left: 63px;"><div data-drawio-colors="color: rgb(0, 0, 0); " style="box-sizing: border-box; font-size: 0px; text-align: center;"><div style="display: inline-block; font-size: 12px; font-family: Helvetica; color: rgb(0, 0, 0); line-height: 1.2; pointer-events: all; white-space: normal; overflow-wrap: normal;"><b>view</b>: validated_sensor (public)</div></div></div></foreignObject><image x="63" y="223.5" width="178" height="17" xlink:href=""/></switch></g></g></g><g data-cell-id="n6nk3BLY6128t3IB6Ma7-7"><g/></g></g></g></g></svg> \ No newline at end of file diff --git a/.docs/redirect.html b/.docs/index.html.tpl similarity index 58% rename from .docs/redirect.html rename to .docs/index.html.tpl index 69ede955cd..d92d7c4b18 100644 --- a/.docs/redirect.html +++ b/.docs/index.html.tpl @@ -1,20 +1,18 @@ <!DOCTYPE html> <html lang="en"> -<!-- This file exists only once - in the final documentation --> <head> <meta charset="UTF-8"> <title>Redirect Notice</title> - <meta http-equiv="Refresh" content="0; url='/infrastructures/dbrepo/1.5/'" /> + <meta http-equiv="Refresh" content="0; url='/infrastructures/dbrepo/${DOC_VERSION}/'" /> </head> <body> <h1>Redirect Notice</h1> <p> - This page should automatically open the documentation for version <code>1.5</code>. In case this page does not load the site is + This page should automatically open the documentation for version <code>${DOC_VERSION}</code>. In case this page does not load the site is available at: </p> <p> - <a href="/infrastructures/dbrepo/1.5/">/infrastructures/dbrepo/1.5/</a> + <a href="/infrastructures/dbrepo/${DOC_VERSION}/">/infrastructures/dbrepo/${DOC_VERSION}/</a> </p> </body> </html> \ No newline at end of file diff --git a/.docs/installation.md b/.docs/installation.md index 60f8a0dd5f..ee0d9b88fa 100644 --- a/.docs/installation.md +++ b/.docs/installation.md @@ -11,7 +11,7 @@ author: Martin Weise If you have [Docker](https://docs.docker.com/engine/install/) already installed on your system, you can install DBRepo with: ```shell -curl -sSL https://gitlab.phaidra.org/fair-data-austria-db-repository/fda-services/-/raw/release-1.5/install.sh | bash +curl -sSL https://gitlab.phaidra.org/fair-data-austria-db-repository/fda-services/-/raw/release-1.6/install.sh | bash ``` !!! bug "Default installation security disclaimer" @@ -38,7 +38,7 @@ SSL/TLS certificate is recommended. Follow the [secure install](#secure-install) Execute the install script to download only the environment and save it to `dist`. ```shell -curl -sSL https://gitlab.phaidra.org/fair-data-austria-db-repository/fda-services/-/raw/release-1.5/install.sh | DOWNLOAD_ONLY=1 bash +curl -sSL https://gitlab.phaidra.org/fair-data-austria-db-repository/fda-services/-/raw/release-1.6/install.sh | DOWNLOAD_ONLY=1 bash ``` ### Static Configuration @@ -80,7 +80,7 @@ the variable `IDENTITY_SERVICE_ADMIN_PASSWORD` in `.env`. Update the client secret of the `dbrepo-client`: ```bash -curl -sSL "https://gitlab.phaidra.org/fair-data-austria-db-repository/fda-services/-/raw/release-1.5/.scripts/reg-client-secret.sh" | bash +curl -sSL "https://gitlab.phaidra.org/fair-data-austria-db-repository/fda-services/-/raw/release-1.6/.scripts/reg-client-secret.sh" | bash ``` Also, update the JWT key according to the diff --git a/.docs/kubernetes.md b/.docs/kubernetes.md index b6fd193fdf..232584ee75 100644 --- a/.docs/kubernetes.md +++ b/.docs/kubernetes.md @@ -6,7 +6,7 @@ author: Martin Weise To install DBRepo in your existing cluster, download the sample [ -`values.yaml`](https://gitlab.phaidra.org/fair-data-austria-db-repository/fda-services/-/blob/release-1.5/helm/dbrepo/values.yaml) +`values.yaml`](https://gitlab.phaidra.org/fair-data-austria-db-repository/fda-services/-/blob/release-1.6/helm/dbrepo/values.yaml) for your deployment and update the variables, especially `hostname`. ```shell diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml index ce79f1a01f..aba94058e0 100644 --- a/.gitlab-ci.yml +++ b/.gitlab-ci.yml @@ -442,7 +442,8 @@ release-docs: - "cp .docs/.swagger/custom.css ./final/${DOC_VERSION}/rest/custom.css" # swagger - "cp -r ./site/* ./final/${DOC_VERSION}" # mkdocs - "cp .docker/dist.tar.gz ./final/${APP_VERSION}/dist.tar.gz" # dist - - "cp .docs/redirect.html ./final/${APP_VERSION}/index.html" # redirect patch docs + - "cp .docs/index.html.tpl ./final/${APP_VERSION}/index.html" # redirect patch docs + - sed -i "s/\${DOC_VERSION}/${DOC_VERSION}/g" index.html - "bash ./.gitlab/gen-badge.sh" - eval $(ssh-agent -s) - "mkdir -p /root/.ssh" @@ -452,7 +453,7 @@ release-docs: - tar czf ./final.tar.gz ./final - "scp -oHostKeyAlgorithms=+ssh-rsa -oPubkeyAcceptedAlgorithms=+ssh-rsa final.tar.gz $CI_DOC_USER@$CI_DOC_IP:final.tar.gz" - "scp -oHostKeyAlgorithms=+ssh-rsa -oPubkeyAcceptedAlgorithms=+ssh-rsa versions.json $CI_DOC_USER@$CI_DOC_IP:/system/user/ifs/infrastructures/public_html/dbrepo/versions.json" - - "scp -oHostKeyAlgorithms=+ssh-rsa -oPubkeyAcceptedAlgorithms=+ssh-rsa .docs/redirect.html $CI_DOC_USER@$CI_DOC_IP:/system/user/ifs/infrastructures/public_html/dbrepo/index.html" + - "scp -oHostKeyAlgorithms=+ssh-rsa -oPubkeyAcceptedAlgorithms=+ssh-rsa .docs/index.html.tpl $CI_DOC_USER@$CI_DOC_IP:/system/user/ifs/infrastructures/public_html/dbrepo/index.html" - 'ssh -oHostKeyAlgorithms=+ssh-rsa -oPubkeyAcceptedAlgorithms=+ssh-rsa $CI_DOC_USER@$CI_DOC_IP "rm -rf /system/user/ifs/infrastructures/public_html/dbrepo/${DOC_VERSION}; tar xzf ./final.tar.gz; rm -f ./final.tar.gz; cp -r ./final/* /system/user/ifs/infrastructures/public_html/dbrepo; rm -rf ./final"' release-libs: diff --git a/.gitlab/gen-badge.sh b/.gitlab/gen-badge.sh index e5bb815915..4124b388d7 100644 --- a/.gitlab/gen-badge.sh +++ b/.gitlab/gen-badge.sh @@ -9,7 +9,7 @@ else echo "[WARNING] Skipping badge generation, displaying default badge text: unknown" fi -curl -fsSL "${CI_SONAR_URL}/api/project_badges/measure?project=${CI_SONAR_PROJECT_KEY}&metric=sqale_rating&token=${CI_SONAR_TOKEN}" > "./final/${DOC_VERSION}/images/maintainability.svg" -curl -fsSL "${CI_SONAR_URL}/api/project_badges/measure?project=${CI_SONAR_PROJECT_KEY}&metric=security_rating&token=${CI_SONAR_TOKEN}" > "./final/${DOC_VERSION}/images/security.svg" +curl -fsSL "${CI_SONAR_URL}/api/project_badges/measure?project=${CI_SONAR_PROJECT_KEY}&metric=sqale_rating&token=${CI_SONAR_BADGE_TOKEN}" > "./final/${DOC_VERSION}/images/maintainability.svg" +curl -fsSL "${CI_SONAR_URL}/api/project_badges/measure?project=${CI_SONAR_PROJECT_KEY}&metric=security_rating&token=${CI_SONAR_BADGE_TOKEN}" > "./final/${DOC_VERSION}/images/security.svg" echo "[INFO] retrieved badges" \ No newline at end of file diff --git a/README.md b/README.md index e039731977..628da3f3f3 100644 --- a/README.md +++ b/README.md @@ -14,7 +14,7 @@ If you have [Docker](https://docs.docker.com/engine/install/) already installed with: ```bash -curl -sSL https://gitlab.phaidra.org/fair-data-austria-db-repository/fda-services/-/raw/release-1.5/install.sh | bash +curl -sSL https://gitlab.phaidra.org/fair-data-austria-db-repository/fda-services/-/raw/release-1.6/install.sh | bash ``` ## Documentation diff --git a/build-docs.sh b/build-docs.sh index fba2afc766..8639b3756f 100644 --- a/build-docs.sh +++ b/build-docs.sh @@ -89,6 +89,6 @@ done # finalization echo "===================================================" echo "Moving HTML redirect and JSON versions to /" -cp ./final/${APP_VERSION}/redirect.html ./final/index.html +cp ./final/${APP_VERSION}/index.html.tpl ./final/index.html cp ./final/${APP_VERSION}/versions.json ./final/versions.json echo "===================================================" diff --git a/dbrepo-dashboard-service/dashboards/system.json b/dbrepo-dashboard-service/dashboards/system.json index edee464f62..e6f81bda40 100644 --- a/dbrepo-dashboard-service/dashboards/system.json +++ b/dbrepo-dashboard-service/dashboards/system.json @@ -18,6 +18,7 @@ "editable": true, "fiscalYearStartMonth": 0, "graphTooltip": 1, + "id": 3, "links": [ { "asDropdown": false, @@ -109,7 +110,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -179,7 +180,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -247,7 +248,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -315,7 +316,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -431,7 +432,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -453,13 +454,83 @@ "title": "Data Volume", "type": "stat" }, + { + "datasource": { + "type": "prometheus", + "uid": "P18F45E9DC7E75912" + }, + "description": "Top 10 by number of accesses", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + } + }, + "mappings": [] + }, + "overrides": [] + }, + "gridPos": { + "h": 7, + "w": 8, + "x": 0, + "y": 4 + }, + "id": 38, + "options": { + "displayLabels": [ + "percent" + ], + "legend": { + "calcs": [], + "displayMode": "hidden", + "placement": "right", + "showLegend": false, + "values": [] + }, + "pieType": "pie", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "targets": [ + { + "datasource": { + "type": "prometheus", + "uid": "P18F45E9DC7E75912" + }, + "editorMode": "code", + "expr": "topk(10, dbrepo_datasource_data_get_total)", + "instant": false, + "legendFormat": "{{uri}}", + "range": true, + "refId": "A" + } + ], + "title": "Popular Data Sources", + "type": "piechart" + }, { "collapsed": false, "gridPos": { "h": 1, "w": 24, "x": 0, - "y": 4 + "y": 11 }, "id": 22, "panels": [], @@ -506,7 +577,7 @@ "h": 3, "w": 4, "x": 0, - "y": 5 + "y": 12 }, "id": 17, "options": { @@ -526,7 +597,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -588,7 +659,7 @@ "h": 3, "w": 4, "x": 4, - "y": 5 + "y": 12 }, "id": 24, "options": { @@ -608,7 +679,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -658,7 +729,7 @@ "h": 3, "w": 4, "x": 8, - "y": 5 + "y": 12 }, "id": 25, "options": { @@ -678,7 +749,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -728,7 +799,7 @@ "h": 3, "w": 4, "x": 12, - "y": 5 + "y": 12 }, "id": 26, "options": { @@ -748,7 +819,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -800,7 +871,7 @@ "h": 3, "w": 4, "x": 16, - "y": 5 + "y": 12 }, "id": 27, "options": { @@ -819,7 +890,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -881,7 +952,7 @@ "h": 7, "w": 12, "x": 0, - "y": 8 + "y": 15 }, "id": 20, "options": { @@ -902,7 +973,7 @@ "sizing": "auto", "valueMode": "color" }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -988,7 +1059,7 @@ "h": 7, "w": 12, "x": 12, - "y": 8 + "y": 15 }, "id": 21, "options": { @@ -1030,7 +1101,7 @@ "h": 1, "w": 24, "x": 0, - "y": 15 + "y": 22 }, "id": 31, "panels": [], @@ -1052,8 +1123,7 @@ "mode": "absolute", "steps": [ { - "color": "purple", - "value": null + "color": "purple" }, { "color": "red", @@ -1081,7 +1151,7 @@ "h": 3, "w": 4, "x": 0, - "y": 16 + "y": 23 }, "id": 32, "options": { @@ -1101,7 +1171,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -1138,8 +1208,7 @@ "mode": "absolute", "steps": [ { - "color": "blue", - "value": null + "color": "blue" } ] }, @@ -1151,7 +1220,7 @@ "h": 3, "w": 4, "x": 4, - "y": 16 + "y": 23 }, "id": 29, "options": { @@ -1171,7 +1240,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -1208,8 +1277,7 @@ "mode": "absolute", "steps": [ { - "color": "blue", - "value": null + "color": "blue" } ] }, @@ -1221,7 +1289,7 @@ "h": 3, "w": 4, "x": 8, - "y": 16 + "y": 23 }, "id": 30, "options": { @@ -1241,7 +1309,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -1278,8 +1346,7 @@ "mode": "absolute", "steps": [ { - "color": "green", - "value": null + "color": "green" }, { "color": "#EAB839", @@ -1303,7 +1370,7 @@ "h": 3, "w": 4, "x": 12, - "y": 16 + "y": 23 }, "id": 35, "options": { @@ -1323,7 +1390,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -1360,8 +1427,7 @@ "mode": "absolute", "steps": [ { - "color": "green", - "value": null + "color": "green" }, { "color": "#EAB839", @@ -1385,7 +1451,7 @@ "h": 3, "w": 4, "x": 16, - "y": 16 + "y": 23 }, "id": 36, "options": { @@ -1405,7 +1471,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -1442,8 +1508,7 @@ "mode": "absolute", "steps": [ { - "color": "green", - "value": null + "color": "green" }, { "color": "#EAB839", @@ -1467,7 +1532,7 @@ "h": 3, "w": 4, "x": 20, - "y": 16 + "y": 23 }, "id": 37, "options": { @@ -1487,7 +1552,7 @@ "textMode": "auto", "wideLayout": true }, - "pluginVersion": "10.4.3", + "pluginVersion": "10.4.9", "targets": [ { "datasource": { @@ -1557,8 +1622,7 @@ "mode": "absolute", "steps": [ { - "color": "green", - "value": null + "color": "green" } ] }, @@ -1601,7 +1665,7 @@ "h": 7, "w": 12, "x": 0, - "y": 19 + "y": 26 }, "id": 33, "options": { @@ -1644,7 +1708,7 @@ "h": 1, "w": 24, "x": 0, - "y": 26 + "y": 33 }, "id": 2, "panels": [], @@ -1699,8 +1763,7 @@ "mode": "absolute", "steps": [ { - "color": "green", - "value": null + "color": "green" }, { "color": "red", @@ -1715,7 +1778,7 @@ "h": 7, "w": 12, "x": 0, - "y": 27 + "y": 34 }, "id": 23, "options": { @@ -1790,8 +1853,7 @@ "mode": "absolute", "steps": [ { - "color": "red", - "value": null + "color": "red" }, { "color": "green", @@ -1806,7 +1868,7 @@ "h": 7, "w": 12, "x": 12, - "y": 27 + "y": 34 }, "id": 16, "options": { @@ -1891,8 +1953,7 @@ "mode": "absolute", "steps": [ { - "color": "green", - "value": null + "color": "green" } ] }, @@ -1965,7 +2026,7 @@ "h": 7, "w": 12, "x": 0, - "y": 34 + "y": 41 }, "id": 6, "options": { @@ -2052,8 +2113,7 @@ "mode": "absolute", "steps": [ { - "color": "green", - "value": null + "color": "green" } ] }, @@ -2126,7 +2186,7 @@ "h": 7, "w": 12, "x": 12, - "y": 34 + "y": 41 }, "id": 7, "options": { @@ -2169,104 +2229,52 @@ "type": "prometheus", "uid": "P18F45E9DC7E75912" }, + "description": "Top 10 by frequency of access", "fieldConfig": { "defaults": { "color": { "mode": "palette-classic" }, "custom": { - "axisBorderShow": false, - "axisCenteredZero": false, - "axisColorMode": "text", - "axisLabel": "", - "axisPlacement": "auto", - "barAlignment": 0, - "drawStyle": "line", - "fillOpacity": 25, - "gradientMode": "none", "hideFrom": { "legend": false, "tooltip": false, "viz": false - }, - "insertNulls": false, - "lineInterpolation": "smooth", - "lineWidth": 2, - "pointSize": 5, - "scaleDistribution": { - "type": "linear" - }, - "showPoints": "auto", - "spanNulls": false, - "stacking": { - "group": "A", - "mode": "none" - }, - "thresholdsStyle": { - "mode": "off" } }, "mappings": [], - "thresholds": { - "mode": "absolute", - "steps": [ - { - "color": "green", - "value": null - } - ] - }, "unit": "reqps" }, - "overrides": [ - { - "matcher": { - "id": "byRegexp", - "options": "/.*search-service.*/" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "orange", - "mode": "fixed" - } - } - ] - }, - { - "matcher": { - "id": "byRegexp", - "options": "/.*analyse-service.*/" - }, - "properties": [ - { - "id": "color", - "value": { - "fixedColor": "super-light-orange", - "mode": "fixed" - } - } - ] - } - ] + "overrides": [] }, "gridPos": { "h": 7, "w": 12, "x": 0, - "y": 41 + "y": 48 }, "id": 18, "options": { + "displayLabels": [ + "percent" + ], "legend": { "calcs": [], - "displayMode": "list", - "placement": "bottom", - "showLegend": true + "displayMode": "hidden", + "placement": "right", + "showLegend": false, + "values": [] + }, + "pieType": "pie", + "reduceOptions": { + "calcs": [ + "lastNotNull" + ], + "fields": "", + "values": false }, "tooltip": { - "mode": "multi", + "mode": "single", "sort": "none" } }, @@ -2277,15 +2285,15 @@ "uid": "P18F45E9DC7E75912" }, "editorMode": "code", - "expr": "rate(flask_http_request_duration_seconds_count{status=~\"200|201|202\",path!=\"/health\"}[$__rate_interval])", + "expr": "topk(10, rate(dbrepo_table_data_get_total[$__range]))", "instant": false, - "legendFormat": "{{method}} {{instance}} {{path}} ({{status}})", + "legendFormat": "__auto", "range": true, "refId": "A" } ], - "title": "Successful API Requests", - "type": "timeseries" + "title": "Popular Datasources", + "type": "piechart" }, { "datasource": { @@ -2334,8 +2342,7 @@ "mode": "absolute", "steps": [ { - "color": "green", - "value": null + "color": "green" } ] }, @@ -2378,7 +2385,7 @@ "h": 7, "w": 12, "x": 12, - "y": 41 + "y": 48 }, "id": 19, "options": { @@ -2421,13 +2428,13 @@ "list": [] }, "time": { - "from": "now-30m", + "from": "now-1h", "to": "now" }, "timepicker": {}, "timezone": "browser", "title": "DBRepo - Overview", "uid": "bdz20owu8zn5se", - "version": 1, + "version": 8, "weekStart": "" } \ No newline at end of file diff --git a/dbrepo-data-service/rest-service/src/main/java/at/tuwien/endpoints/SubsetEndpoint.java b/dbrepo-data-service/rest-service/src/main/java/at/tuwien/endpoints/SubsetEndpoint.java index 13cfc5c560..4f1b5d59c9 100644 --- a/dbrepo-data-service/rest-service/src/main/java/at/tuwien/endpoints/SubsetEndpoint.java +++ b/dbrepo-data-service/rest-service/src/main/java/at/tuwien/endpoints/SubsetEndpoint.java @@ -10,10 +10,7 @@ import at.tuwien.api.database.query.QueryPersistDto; import at.tuwien.api.error.ApiErrorDto; import at.tuwien.exception.*; import at.tuwien.mapper.MetadataMapper; -import at.tuwien.service.CredentialService; -import at.tuwien.service.SchemaService; -import at.tuwien.service.StorageService; -import at.tuwien.service.SubsetService; +import at.tuwien.service.*; import at.tuwien.utils.UserUtil; import at.tuwien.validation.EndpointValidator; import io.micrometer.observation.annotation.Observed; @@ -55,17 +52,19 @@ public class SubsetEndpoint extends AbstractEndpoint { private final SchemaService schemaService; private final SubsetService subsetService; private final MetadataMapper metadataMapper; + private final MetricsService metricsService; private final StorageService storageService; private final CredentialService credentialService; private final EndpointValidator endpointValidator; @Autowired public SubsetEndpoint(SchemaService schemaService, SubsetService subsetService, MetadataMapper metadataMapper, - StorageService storageService, CredentialService credentialService, - EndpointValidator endpointValidator) { + MetricsService metricsService, StorageService storageService, + CredentialService credentialService, EndpointValidator endpointValidator) { this.schemaService = schemaService; this.subsetService = subsetService; this.metadataMapper = metadataMapper; + this.metricsService = metricsService; this.storageService = storageService; this.credentialService = credentialService; this.endpointValidator = endpointValidator; @@ -188,6 +187,7 @@ public class SubsetEndpoint extends AbstractEndpoint { log.trace("accept header matches csv"); try { final Dataset<Row> dataset = subsetService.getData(database, subset, null, null); + metricsService.countSubsetGetData(databaseId, subsetId); final ExportResourceDto resource = storageService.transformDataset(dataset); final HttpHeaders headers = new HttpHeaders(); headers.add("Content-Disposition", "attachment; filename=\"" + resource.getFilename() + "\""); @@ -366,6 +366,7 @@ public class SubsetEndpoint extends AbstractEndpoint { .build(); } final Dataset<Row> dataset = subsetService.getData(database, subset, page, size); + metricsService.countSubsetGetData(databaseId, subsetId); final ViewDto view = schemaService.inspectView(database, metadataMapper.queryDtoToViewName(subset)); headers.set("Access-Control-Expose-Headers", "X-Id X-Headers"); headers.set("X-Headers", String.join(",", view.getColumns().stream().map(ViewColumnDto::getInternalName).toList())); diff --git a/dbrepo-data-service/rest-service/src/main/java/at/tuwien/endpoints/TableEndpoint.java b/dbrepo-data-service/rest-service/src/main/java/at/tuwien/endpoints/TableEndpoint.java index 4191726dd9..11720b4906 100644 --- a/dbrepo-data-service/rest-service/src/main/java/at/tuwien/endpoints/TableEndpoint.java +++ b/dbrepo-data-service/rest-service/src/main/java/at/tuwien/endpoints/TableEndpoint.java @@ -12,10 +12,7 @@ import at.tuwien.api.database.table.internal.TableCreateDto; import at.tuwien.api.error.ApiErrorDto; import at.tuwien.exception.*; import at.tuwien.gateway.MetadataServiceGateway; -import at.tuwien.service.CredentialService; -import at.tuwien.service.SchemaService; -import at.tuwien.service.StorageService; -import at.tuwien.service.TableService; +import at.tuwien.service.*; import at.tuwien.utils.UserUtil; import at.tuwien.validation.EndpointValidator; import io.micrometer.observation.annotation.Observed; @@ -55,17 +52,19 @@ public class TableEndpoint extends AbstractEndpoint { private final TableService tableService; private final SchemaService schemaService; + private final MetricsService metricsService; private final StorageService storageService; private final CredentialService credentialService; private final EndpointValidator endpointValidator; private final MetadataServiceGateway metadataServiceGateway; @Autowired - public TableEndpoint(TableService tableService, SchemaService schemaService, StorageService storageService, - CredentialService credentialService, EndpointValidator endpointValidator, - MetadataServiceGateway metadataServiceGateway) { + public TableEndpoint(TableService tableService, SchemaService schemaService, MetricsService metricsService, + StorageService storageService, CredentialService credentialService, + EndpointValidator endpointValidator, MetadataServiceGateway metadataServiceGateway) { this.tableService = tableService; this.schemaService = schemaService; + this.metricsService = metricsService; this.storageService = storageService; this.credentialService = credentialService; this.endpointValidator = endpointValidator; @@ -291,10 +290,12 @@ public class TableEndpoint extends AbstractEndpoint { } headers.set("Access-Control-Expose-Headers", "X-Headers"); headers.set("X-Headers", String.join(",", table.getColumns().stream().map(ColumnDto::getInternalName).toList())); + final Dataset<Row> dataset = tableService.getData(table.getDatabase(), table.getInternalName(), timestamp, + null, null, null, null); + metricsService.countTableGetData(databaseId, tableId); return ResponseEntity.ok() .headers(headers) - .body(transform(tableService.getData(table.getDatabase(), table.getInternalName(), timestamp, - null, null, null, null))); + .body(transform(dataset)); } catch (SQLException | QueryMalformedException e) { log.error("Failed to establish connection to database: {}", e.getMessage()); throw new DatabaseUnavailableException("Failed to establish connection to database: " + e.getMessage(), e); @@ -625,6 +626,7 @@ public class TableEndpoint extends AbstractEndpoint { } final Dataset<Row> dataset = tableService.getData(table.getDatabase(), table.getInternalName(), timestamp, null, null, null, null); + metricsService.countTableGetData(databaseId, tableId); final ExportResourceDto resource = storageService.transformDataset(dataset); final HttpHeaders headers = new HttpHeaders(); headers.add("Content-Disposition", "attachment; filename=\"" + resource.getFilename() + "\""); diff --git a/dbrepo-data-service/rest-service/src/main/java/at/tuwien/endpoints/ViewEndpoint.java b/dbrepo-data-service/rest-service/src/main/java/at/tuwien/endpoints/ViewEndpoint.java index b08c300b45..c8da423986 100644 --- a/dbrepo-data-service/rest-service/src/main/java/at/tuwien/endpoints/ViewEndpoint.java +++ b/dbrepo-data-service/rest-service/src/main/java/at/tuwien/endpoints/ViewEndpoint.java @@ -8,10 +8,7 @@ import at.tuwien.api.database.internal.PrivilegedDatabaseDto; import at.tuwien.api.database.internal.PrivilegedViewDto; import at.tuwien.api.error.ApiErrorDto; import at.tuwien.exception.*; -import at.tuwien.service.CredentialService; -import at.tuwien.service.StorageService; -import at.tuwien.service.TableService; -import at.tuwien.service.ViewService; +import at.tuwien.service.*; import at.tuwien.utils.UserUtil; import at.tuwien.validation.EndpointValidator; import io.micrometer.observation.annotation.Observed; @@ -26,6 +23,8 @@ import jakarta.servlet.http.HttpServletRequest; import jakarta.validation.Valid; import jakarta.validation.constraints.NotNull; import lombok.extern.log4j.Log4j2; +import org.apache.spark.sql.Dataset; +import org.apache.spark.sql.Row; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.core.io.InputStreamResource; import org.springframework.http.HttpHeaders; @@ -48,15 +47,18 @@ public class ViewEndpoint extends AbstractEndpoint { private final ViewService viewService; private final TableService tableService; + private final MetricsService metricsService; private final StorageService storageService; private final CredentialService credentialService; private final EndpointValidator endpointValidator; @Autowired - public ViewEndpoint(ViewService viewService, TableService tableService, StorageService storageService, - CredentialService credentialService, EndpointValidator endpointValidator) { + public ViewEndpoint(ViewService viewService, TableService tableService, MetricsService metricsService, + StorageService storageService, CredentialService credentialService, + EndpointValidator endpointValidator) { this.viewService = viewService; this.tableService = tableService; + this.metricsService = metricsService; this.storageService = storageService; this.credentialService = credentialService; this.endpointValidator = endpointValidator; @@ -286,10 +288,12 @@ public class ViewEndpoint extends AbstractEndpoint { } headers.set("Access-Control-Expose-Headers", "X-Headers"); headers.set("X-Headers", String.join(",", view.getColumns().stream().map(ViewColumnDto::getInternalName).toList())); + final Dataset<Row> dataset = tableService.getData(view.getDatabase(), view.getInternalName(), timestamp, + page, size, null, null); + metricsService.countViewGetData(databaseId, viewId); return ResponseEntity.ok() .headers(headers) - .body(transform(tableService.getData(view.getDatabase(), view.getInternalName(), timestamp, page, - size, null, null))); + .body(transform(dataset)); } catch (SQLException e) { log.error("Failed to establish connection to database: {}", e.getMessage()); throw new DatabaseUnavailableException("Failed to establish connection to database: " + e.getMessage(), e); @@ -349,9 +353,11 @@ public class ViewEndpoint extends AbstractEndpoint { } credentialService.getAccess(databaseId, UserUtil.getId(principal)); } + final Dataset<Row> dataset = tableService.getData(view.getDatabase(), view.getInternalName(), timestamp, null, + null, null, null); + metricsService.countViewGetData(databaseId, viewId); + final ExportResourceDto resource = storageService.transformDataset(dataset); final HttpHeaders headers = new HttpHeaders(); - final ExportResourceDto resource = storageService.transformDataset(tableService.getData(view.getDatabase(), - view.getInternalName(), timestamp, null, null, null, null)); headers.add("Content-Disposition", "attachment; filename=\"" + resource.getFilename() + "\""); log.trace("export table resulted in resource {}", resource); return ResponseEntity.ok() diff --git a/dbrepo-data-service/rest-service/src/main/resources/application.yml b/dbrepo-data-service/rest-service/src/main/resources/application.yml index 9848928b7b..03d89895cb 100644 --- a/dbrepo-data-service/rest-service/src/main/resources/application.yml +++ b/dbrepo-data-service/rest-service/src/main/resources/application.yml @@ -82,3 +82,4 @@ dbrepo: exchangeName: "${BROKER_EXCHANGE_NAME:dbrepo}" routingKey: "${BROKER_ROUTING_KEY:#}" connectionTimeout: "${SPARQL_CONNECTION_TIMEOUT:10000}" + baseUrl: "${BASE_URL:http://localhost}" diff --git a/dbrepo-data-service/services/src/main/java/at/tuwien/config/CacheConfig.java b/dbrepo-data-service/services/src/main/java/at/tuwien/config/CacheConfig.java index a77af353d3..45654157d1 100644 --- a/dbrepo-data-service/services/src/main/java/at/tuwien/config/CacheConfig.java +++ b/dbrepo-data-service/services/src/main/java/at/tuwien/config/CacheConfig.java @@ -1,25 +1,64 @@ package at.tuwien.config; -import at.tuwien.api.PrivilegedObjectDto; +import at.tuwien.api.container.internal.PrivilegedContainerDto; +import at.tuwien.api.database.DatabaseAccessDto; +import at.tuwien.api.database.internal.PrivilegedDatabaseDto; +import at.tuwien.api.database.internal.PrivilegedViewDto; +import at.tuwien.api.database.table.internal.PrivilegedTableDto; +import at.tuwien.api.user.internal.PrivilegedUserDto; import com.github.benmanes.caffeine.cache.Cache; import com.github.benmanes.caffeine.cache.Caffeine; import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +import java.util.UUID; import java.util.concurrent.TimeUnit; @Configuration -public class CacheConfig<K, T extends PrivilegedObjectDto> { +public class CacheConfig { @Value("${dbrepo.credentialCacheTimeout}") private Long credentialCacheTimeout; @Bean - public Cache<K, T> cache() { - return Caffeine.newBuilder() - .expireAfterWrite(credentialCacheTimeout, TimeUnit.SECONDS) - .build(); + public Cache<UUID, PrivilegedUserDto> userCache() { + return new ExpiryCache<UUID, PrivilegedUserDto>().build(); + } + + @Bean + public Cache<Long, PrivilegedViewDto> viewCache() { + return new ExpiryCache<Long, PrivilegedViewDto>().build(); + } + + @Bean + public Cache<Long, DatabaseAccessDto> accessCache() { + return new ExpiryCache<Long, DatabaseAccessDto>().build(); + } + + @Bean + public Cache<Long, PrivilegedTableDto> tableCache() { + return new ExpiryCache<Long, PrivilegedTableDto>().build(); + } + + @Bean + public Cache<Long, PrivilegedDatabaseDto> databaseCache() { + return new ExpiryCache<Long, PrivilegedDatabaseDto>().build(); + } + + @Bean + public Cache<Long, PrivilegedContainerDto> containerCache() { + return new ExpiryCache<Long, PrivilegedContainerDto>().build(); + } + + class ExpiryCache<K, T> { + + Cache<K, T> build() { + return Caffeine.newBuilder() + .expireAfterWrite(credentialCacheTimeout, TimeUnit.SECONDS) + .build(); + } + } } diff --git a/dbrepo-data-service/services/src/main/java/at/tuwien/config/MetricsConfig.java b/dbrepo-data-service/services/src/main/java/at/tuwien/config/MetricsConfig.java index 9ff09ab42b..af9ea49f9f 100644 --- a/dbrepo-data-service/services/src/main/java/at/tuwien/config/MetricsConfig.java +++ b/dbrepo-data-service/services/src/main/java/at/tuwien/config/MetricsConfig.java @@ -1,33 +1,21 @@ package at.tuwien.config; -import io.micrometer.core.instrument.Counter; -import io.micrometer.core.instrument.Metrics; import io.micrometer.observation.ObservationRegistry; import io.micrometer.observation.aop.ObservedAspect; +import lombok.Getter; +import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; +@Getter @Configuration public class MetricsConfig { + @Value("${dbrepo.baseUrl}") + private String baseUrl; + @Bean public ObservedAspect observedAspect(ObservationRegistry observationRegistry) { return new ObservedAspect(observationRegistry); } - - @Bean - public Counter httpDataAccessCounter() { - return Counter.builder("dbrepo.data.access") - .tag("protocol", "http") - .description("The total number of accessed data sources") - .register(Metrics.globalRegistry); - } - - @Bean - public Counter amqpDataAccessCounter() { - return Counter.builder("dbrepo.data.access") - .tag("protocol", "amqp") - .description("The total number of accessed data sources") - .register(Metrics.globalRegistry); - } } diff --git a/dbrepo-data-service/services/src/main/java/at/tuwien/gateway/impl/KeycloakGatewayImpl.java b/dbrepo-data-service/services/src/main/java/at/tuwien/gateway/impl/KeycloakGatewayImpl.java index fe1b2e3611..1e235de4c6 100644 --- a/dbrepo-data-service/services/src/main/java/at/tuwien/gateway/impl/KeycloakGatewayImpl.java +++ b/dbrepo-data-service/services/src/main/java/at/tuwien/gateway/impl/KeycloakGatewayImpl.java @@ -1,9 +1,10 @@ package at.tuwien.gateway.impl; -import at.tuwien.api.auth.KeycloakErrorDto; import at.tuwien.api.keycloak.TokenDto; import at.tuwien.config.KeycloakConfig; -import at.tuwien.exception.*; +import at.tuwien.exception.AccountNotSetupException; +import at.tuwien.exception.AuthServiceConnectionException; +import at.tuwien.exception.CredentialsInvalidException; import at.tuwien.gateway.KeycloakGateway; import lombok.extern.log4j.Log4j2; import org.springframework.beans.factory.annotation.Autowired; @@ -40,18 +41,6 @@ public class KeycloakGatewayImpl implements KeycloakGateway { payload.add("client_secret", keycloakConfig.getKeycloakClientSecret()); final String url = keycloakConfig.getKeycloakEndpoint() + "/realms/dbrepo/protocol/openid-connect/token"; log.trace("request user token from url: {}", url); - log.trace("request username: {}", username); - if (password.isEmpty() || password.isBlank()) { - log.warn("request password: (empty)"); - } else { - log.trace("request password: (set)"); - } - log.trace("request client_id: {}", keycloakConfig.getKeycloakClient()); - if (keycloakConfig.getKeycloakClientSecret().isEmpty() || keycloakConfig.getKeycloakClientSecret().isBlank()) { - log.warn("request client_secret: (empty)"); - } else { - log.trace("request client_secret: (set)"); - } final ResponseEntity<TokenDto> response; try { response = new RestTemplate() diff --git a/dbrepo-data-service/services/src/main/java/at/tuwien/mapper/MetadataMapper.java b/dbrepo-data-service/services/src/main/java/at/tuwien/mapper/MetadataMapper.java index 884732bdc1..0adfafa8f9 100644 --- a/dbrepo-data-service/services/src/main/java/at/tuwien/mapper/MetadataMapper.java +++ b/dbrepo-data-service/services/src/main/java/at/tuwien/mapper/MetadataMapper.java @@ -64,4 +64,22 @@ public interface MetadataMapper { IdentifierBriefDto identifierDtoToIdentifierBriefDto(IdentifierDto data); + default String metricToUri(String baseUrl, Long databaseId, Long tableId, Long subsetId, Long viewId) { + final StringBuilder uri = new StringBuilder(baseUrl) + .append("/database/") + .append(databaseId); + if (tableId != null) { + uri.append("/table/") + .append(tableId); + } else if (subsetId != null) { + uri.append("/subset/") + .append(subsetId); + } else if (viewId != null) { + uri.append("/view/") + .append(viewId); + } + log.trace("count uri: {}", uri); + return uri.toString(); + } + } diff --git a/dbrepo-data-service/services/src/main/java/at/tuwien/service/MetricsService.java b/dbrepo-data-service/services/src/main/java/at/tuwien/service/MetricsService.java new file mode 100644 index 0000000000..131bae7287 --- /dev/null +++ b/dbrepo-data-service/services/src/main/java/at/tuwien/service/MetricsService.java @@ -0,0 +1,10 @@ +package at.tuwien.service; + +public interface MetricsService { + + void countTableGetData(Long databaseId, Long tableId); + + void countSubsetGetData(Long databaseId, Long subsetId); + + void countViewGetData(Long databaseId, Long viewId); +} diff --git a/dbrepo-data-service/services/src/main/java/at/tuwien/service/impl/MetricsServicePrometheusImpl.java b/dbrepo-data-service/services/src/main/java/at/tuwien/service/impl/MetricsServicePrometheusImpl.java new file mode 100644 index 0000000000..73754c9f5f --- /dev/null +++ b/dbrepo-data-service/services/src/main/java/at/tuwien/service/impl/MetricsServicePrometheusImpl.java @@ -0,0 +1,47 @@ +package at.tuwien.service.impl; + +import at.tuwien.config.MetricsConfig; +import at.tuwien.mapper.MetadataMapper; +import at.tuwien.service.MetricsService; +import io.micrometer.core.instrument.Counter; +import io.micrometer.core.instrument.Metrics; +import lombok.extern.log4j.Log4j2; +import org.springframework.stereotype.Service; + +@Log4j2 +@Service +public class MetricsServicePrometheusImpl implements MetricsService { + + private final MetricsConfig metricsConfig; + private final MetadataMapper metadataMapper; + + public MetricsServicePrometheusImpl(MetricsConfig metricsConfig, MetadataMapper metadataMapper) { + this.metricsConfig = metricsConfig; + this.metadataMapper = metadataMapper; + } + + @Override + public void countTableGetData(Long databaseId, Long tableId) { + countGetData(databaseId, tableId, null, null); + } + + @Override + public void countSubsetGetData(Long databaseId, Long subsetId) { + countGetData(databaseId, null, subsetId, null); + } + + @Override + public void countViewGetData(Long databaseId, Long viewId) { + countGetData(databaseId, null, null, viewId); + } + + public void countGetData(Long databaseId, Long tableId, Long subsetId, Long viewId) { + Counter.builder("dbrepo.datasource.data.get") + .tag("uri", metadataMapper.metricToUri(metricsConfig.getBaseUrl(), databaseId, tableId, subsetId, viewId)) + .tag("protocol", "http") + .description("The total number of accessed data sources") + .register(Metrics.globalRegistry) + .increment(); + } + +} diff --git a/dbrepo-data-service/services/src/main/java/at/tuwien/service/impl/QueueServiceRabbitMqImpl.java b/dbrepo-data-service/services/src/main/java/at/tuwien/service/impl/QueueServiceRabbitMqImpl.java index aff8513787..d5127e050e 100644 --- a/dbrepo-data-service/services/src/main/java/at/tuwien/service/impl/QueueServiceRabbitMqImpl.java +++ b/dbrepo-data-service/services/src/main/java/at/tuwien/service/impl/QueueServiceRabbitMqImpl.java @@ -6,7 +6,6 @@ import at.tuwien.mapper.DataMapper; import at.tuwien.mapper.MetadataMapper; import at.tuwien.service.QueueService; import com.mchange.v2.c3p0.ComboPooledDataSource; -import io.micrometer.core.instrument.Counter; import lombok.extern.log4j.Log4j2; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; @@ -21,14 +20,11 @@ import java.util.Optional; @Service public class QueueServiceRabbitMqImpl extends HibernateConnector implements QueueService { - private final Counter amqpDataAccessCounter; private final DataMapper dataMapper; private final MetadataMapper metadataMapper; @Autowired - public QueueServiceRabbitMqImpl(Counter amqpDataAccessCounter, DataMapper dataMapper, - MetadataMapper metadataMapper) { - this.amqpDataAccessCounter = amqpDataAccessCounter; + public QueueServiceRabbitMqImpl(DataMapper dataMapper, MetadataMapper metadataMapper) { this.dataMapper = dataMapper; this.metadataMapper = metadataMapper; } @@ -54,7 +50,6 @@ public class QueueServiceRabbitMqImpl extends HibernateConnector implements Queu preparedStatement.executeUpdate(); log.trace("executed statement in {} ms", System.currentTimeMillis() - start); log.trace("successfully inserted tuple"); - amqpDataAccessCounter.increment(); } finally { dataSource.close(); } diff --git a/dbrepo-metadata-service/repositories/src/main/java/at/tuwien/mapper/MetadataMapper.java b/dbrepo-metadata-service/repositories/src/main/java/at/tuwien/mapper/MetadataMapper.java index 98c77c7b0d..f63c387c98 100644 --- a/dbrepo-metadata-service/repositories/src/main/java/at/tuwien/mapper/MetadataMapper.java +++ b/dbrepo-metadata-service/repositories/src/main/java/at/tuwien/mapper/MetadataMapper.java @@ -483,7 +483,6 @@ public interface MetadataMapper { @Mappings({ @Mapping(target = "databaseId", source = "tdbid"), - @Mapping(target = "isPublic", source = "database.isPublic"), }) TableBriefDto tableToTableBriefDto(Table data); @@ -535,7 +534,7 @@ public interface MetadataMapper { .internalName(data.getInternalName()) .owner(userToUserBriefDto(data.getOwner())) .tdbid(data.getTdbid()) - .isPublic(data.getDatabase().getIsPublic()) + .isPublic(data.getIsPublic()) .isSchemaPublic(data.getIsSchemaPublic()) .isVersioned(true) .description(data.getDescription()) @@ -720,7 +719,7 @@ public interface MetadataMapper { @Mappings({ @Mapping(target = "tableId", source = "table.id"), @Mapping(target = "databaseId", source = "table.database.id"), - @Mapping(target = "isPublic", source = "table.database.isPublic"), + @Mapping(target = "isPublic", source = "table.isSchemaPublic"), @Mapping(target = "description", source = "description"), @Mapping(target = "table", ignore = true), @Mapping(target = "views", ignore = true) diff --git a/dbrepo-metadata-service/rest-service/src/main/java/at/tuwien/endpoints/TableEndpoint.java b/dbrepo-metadata-service/rest-service/src/main/java/at/tuwien/endpoints/TableEndpoint.java index 3dbc450735..a517c39d10 100644 --- a/dbrepo-metadata-service/rest-service/src/main/java/at/tuwien/endpoints/TableEndpoint.java +++ b/dbrepo-metadata-service/rest-service/src/main/java/at/tuwien/endpoints/TableEndpoint.java @@ -403,7 +403,8 @@ public class TableEndpoint { @NotNull Principal principal) throws NotAllowedException, DataServiceException, DataServiceConnectionException, DatabaseNotFoundException, TableNotFoundException, SearchServiceException, SearchServiceConnectionException { - log.debug("endpoint update table, databaseId={}, data.is_public={}", databaseId, data.getIsPublic()); + log.debug("endpoint update table, databaseId={}, data.is_public={}, data.is_schema_public={}", databaseId, + data.getIsPublic(), data.getIsSchemaPublic()); final Table table = tableService.findById(databaseId, tableId); if (!table.getOwner().equals(principal)) { log.error("Failed to update table: not owner"); diff --git a/dbrepo-metadata-service/services/src/main/java/at/tuwien/service/impl/TableServiceImpl.java b/dbrepo-metadata-service/services/src/main/java/at/tuwien/service/impl/TableServiceImpl.java index 4a4a9ccaca..57b546fcea 100644 --- a/dbrepo-metadata-service/services/src/main/java/at/tuwien/service/impl/TableServiceImpl.java +++ b/dbrepo-metadata-service/services/src/main/java/at/tuwien/service/impl/TableServiceImpl.java @@ -220,8 +220,8 @@ public class TableServiceImpl implements TableService { } final Table tableEntity = optional.get(); tableEntity.setIsPublic(data.getIsPublic()); - tableEntity.setDescription(data.getDescription()); tableEntity.setIsSchemaPublic(data.getIsSchemaPublic()); + tableEntity.setDescription(data.getDescription()); final Database database = databaseRepository.save(table.getDatabase()); /* update in search service */ searchServiceGateway.update(database); diff --git a/dbrepo-ui/components/subset/SubsetList.vue b/dbrepo-ui/components/subset/SubsetList.vue index b977daffa5..f57dc68a88 100644 --- a/dbrepo-ui/components/subset/SubsetList.vue +++ b/dbrepo-ui/components/subset/SubsetList.vue @@ -26,6 +26,21 @@ :to="link(item)" :href="link(item)"> <template v-slot:append> + <v-chip + v-if="database.is_public" + size="small" + class="ml-2" + color="success" + :text="$t('toolbars.database.public')" + variant="outlined" /> + <v-chip + v-if="!database.is_public" + size="small" + class="ml-2" + :color="colorVariant" + variant="outlined" + :text="$t('toolbars.database.private')" + flat /> <v-tooltip v-if="hasPublishedIdentifier(item)" :text="$t('pages.identifier.pid.title')" @@ -65,6 +80,15 @@ export default { }, database () { return this.cacheStore.getDatabase + }, + isContrastTheme () { + return this.$vuetify.theme.global.name.toLowerCase().endsWith('contrast') + }, + isDarkTheme () { + return this.$vuetify.theme.global.name.toLowerCase().startsWith('dark') + }, + colorVariant () { + return this.isContrastTheme ? '' : (this.isDarkTheme ? 'tertiary' : 'secondary') } }, mounted () { @@ -88,25 +112,25 @@ export default { toast.error(this.$t(code)) }) }, - title (query) { - if (query.identifiers.length === 0) { - return formatTimestampUTCLabel(query.created) + title (subset) { + if (subset.identifiers.length === 0) { + return subset.query } const identifierService = useIdentifierService() - return identifierService.identifierPreferEnglishTitle(query.identifiers[0]) + return identifierService.identifierPreferEnglishTitle(subset.identifiers[0]) }, - subtitle (query) { - if (query.identifiers.length === 0) { + subtitle (subset) { + if (subset.identifiers.length === 0) { return null } const identifierService = useIdentifierService() - return identifierService.identifierPreferEnglishDescription(query.identifiers[0]) + return identifierService.identifierPreferEnglishDescription(subset.identifiers[0]) }, - link (query) { - return `/database/${this.$route.params.database_id}/subset/${query.id}/info` + link (subset) { + return `/database/${this.$route.params.database_id}/subset/${subset.id}/info` }, - clazz (view) { - return this.hasPublishedIdentifier(view) ? 'primary-text' : null + clazz (subset) { + return this.hasPublishedIdentifier(subset) ? 'primary-text' : null }, hasPublishedIdentifier (subset) { if (!subset.identifiers) { diff --git a/dbrepo-ui/components/view/ViewList.vue b/dbrepo-ui/components/view/ViewList.vue index 6fe8451903..543a8746af 100644 --- a/dbrepo-ui/components/view/ViewList.vue +++ b/dbrepo-ui/components/view/ViewList.vue @@ -61,9 +61,6 @@ export default { } }, computed: { - loadingColor () { - return this.error ? 'red lighten-2' : 'primary' - }, user () { return this.userStore.getUser }, diff --git a/dbrepo-ui/components/view/ViewToolbar.vue b/dbrepo-ui/components/view/ViewToolbar.vue index 1f122cc7a4..64ea3f1029 100644 --- a/dbrepo-ui/components/view/ViewToolbar.vue +++ b/dbrepo-ui/components/view/ViewToolbar.vue @@ -187,7 +187,7 @@ export default { return this.access.type === 'read' || this.access.type === 'write_own' || this.access.type === 'write_all' }, canReadData () { - if (!this.view) { + if (!this.cachedView) { return false } if (this.cachedView.is_public) { diff --git a/dbrepo-ui/pages/database/[database_id]/view/[view_id]/data.vue b/dbrepo-ui/pages/database/[database_id]/view/[view_id]/data.vue index 6aed9307c4..9751fce5b7 100644 --- a/dbrepo-ui/pages/database/[database_id]/view/[view_id]/data.vue +++ b/dbrepo-ui/pages/database/[database_id]/view/[view_id]/data.vue @@ -96,7 +96,7 @@ export default { return this.access.type === 'read' || this.access.type === 'write_own' || this.access.type === 'write_all' }, canReadData () { - if (!this.view) { + if (!this.cachedView) { return false } if (this.cachedView.is_public) { diff --git a/helm/dbrepo/README.md b/helm/dbrepo/README.md index f8c6b12d75..b995791a15 100644 --- a/helm/dbrepo/README.md +++ b/helm/dbrepo/README.md @@ -1,13 +1,13 @@ # DBRepo Helm chart -[DBRepo](https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.5/) is a database repository system that +[DBRepo](https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.6/) is a database repository system that allows researchers to ingest data into a central, versioned repository through common interfaces. ## TL;DR Download the sample [ -`values.yaml`](https://gitlab.phaidra.org/fair-data-austria-db-repository/fda-services/-/raw/release-1.5/helm-charts/dbrepo/values.yaml?inline=true) +`values.yaml`](https://gitlab.phaidra.org/fair-data-austria-db-repository/fda-services/-/raw/release-1.6/helm-charts/dbrepo/values.yaml?inline=true) for your deployment and update the variables, especially `hostname`. ```bash @@ -65,10 +65,10 @@ The command removes all the Kubernetes components associated with the chart and ### Metadata Database | Name | Description | Value | -| ---------------------------------------- | -------------------------------------------------------------------------------------------------------------------------------------- | ---------------------------------------------------------------------- | +| ---------------------------------------- |----------------------------------------------------------------------------------------------------------------------------------------| ---------------------------------------------------------------------- | | `metadatadb.enabled` | Enable the Metadata datadb. | `true` | | `metadatadb.host` | The hostname for the microservices. | `metadata-db` | -| `metadatadb.extraFlags` | Extra flags to ensure the query store works as intended, ref https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.5/api/data-db/#data | `--character-set-server=utf8mb4 --collation-server=utf8mb4_general_ci` | +| `metadatadb.extraFlags` | Extra flags to ensure the query store works as intended, ref https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.6/api/data-db/#data | `--character-set-server=utf8mb4 --collation-server=utf8mb4_general_ci` | | `metadatadb.rootUser.user` | The root username. | `root` | | `metadatadb.rootUser.password` | The root user password. | `dbrepo` | | `metadatadb.db.name` | The database name. | `dbrepo` | @@ -96,9 +96,9 @@ The command removes all the Kubernetes components associated with the chart and ### Data Database | Name | Description | Value | -| ------------------------------------ | -------------------------------------------------------------------------------------------------------------------------------------- | ---------------------------------------------------------------------- | +| ------------------------------------ |----------------------------------------------------------------------------------------------------------------------------------------| ---------------------------------------------------------------------- | | `datadb.host` | The hostname for the microservices. | `data-db` | -| `datadb.extraFlags` | Extra flags to ensure the query store works as intended, ref https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.5/api/data-db/#data | `--character-set-server=utf8mb4 --collation-server=utf8mb4_general_ci` | +| `datadb.extraFlags` | Extra flags to ensure the query store works as intended, ref https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.6/api/data-db/#data | `--character-set-server=utf8mb4 --collation-server=utf8mb4_general_ci` | | `datadb.rootUser.user` | The root username. | `root` | | `datadb.rootUser.password` | The root user password. | `dbrepo` | | `datadb.db.name` | The database name. | `dbrepo` | diff --git a/lib/python/README.md b/lib/python/README.md index 7b5b7d6da7..443787cbb4 100644 --- a/lib/python/README.md +++ b/lib/python/README.md @@ -48,17 +48,17 @@ client.import_table_data(database_id=7, table_id=13, file_name_or_data_frame=df) ## Supported Features & Best-Practices - Manage user - account ([docs](https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.5/api/#create-user-account)) + account ([docs](https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.6/api/#create-user-account)) - Manage - databases ([docs](https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.5/usage-overview/#create-database)) + databases ([docs](https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.6/usage-overview/#create-database)) - Manage database access & - visibility ([docs](https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.5/api/#create-database)) + visibility ([docs](https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.6/api/#create-database)) - Import - dataset ([docs](https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.5/api/#import-dataset)) + dataset ([docs](https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.6/api/#import-dataset)) - Create persistent - identifiers ([docs](https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.5/api/#assign-database-pid)) + identifiers ([docs](https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.6/api/#assign-database-pid)) - Execute - queries ([docs](https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.5/api/#export-subset)) + queries ([docs](https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.6/api/#export-subset)) - Get data from tables/views/subsets ## Configure diff --git a/mkdocs.yml b/mkdocs.yml index b8e3ed5a6d..5ec3a92799 100644 --- a/mkdocs.yml +++ b/mkdocs.yml @@ -1,5 +1,5 @@ site_name: Database Repository -site_url: https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.5/ +site_url: https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.6/ repo_url: https://gitlab.phaidra.org/fair-data-austria-db-repository/fda-services repo_name: fda-services site_author: Research Unit Data Science, Technische Universität Wien @@ -17,6 +17,8 @@ nav: - Concepts: - Overview: concepts/index.md - Authentication: concepts/authentication.md + - Data Versioning: concepts/data-versioning.md + - Data Visibility: concepts/data-visibility.md - Messaging: concepts/messaging.md - Monitoring: concepts/monitoring.md - Persistent Identifier: concepts/pid.md @@ -118,9 +120,9 @@ markdown_extensions: custom_icons: - .docs/overrides/.icons extra: - homepage: https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/ + homepage: https://www.ifs.tuwien.ac.at/infrastructures/dbrepo/1.6/ version: - default: 1.5 + default: 1.6 provider: mike social: - icon: simple/artifacthub diff --git a/versions.json b/versions.json index 8ba789267d..b42a3dfd7d 100644 --- a/versions.json +++ b/versions.json @@ -1,4 +1,9 @@ [ + { + "version": "1.6", + "title": "1.6", + "aliases": [] + }, { "version": "1.5", "title": "1.5", -- GitLab