Skip to content
Snippets Groups Projects
Unverified Commit a2390123 authored by xgui3783's avatar xgui3783 Committed by GitHub
Browse files

Merge pull request #1071 from FZJ-INM1-BDA/chore_regionMisc

chore/feat: fix region description
parents c7a964f1 3ad97f1d
No related branches found
No related tags found
No related merge requests found
Showing
with 467 additions and 158 deletions
...@@ -6,16 +6,16 @@ jobs: ...@@ -6,16 +6,16 @@ jobs:
build-docker-img: build-docker-img:
runs-on: ubuntu-latest runs-on: ubuntu-latest
env: env:
MATOMO_ID_DEV: '7'
MATOMO_URL_DEV: 'https://stats-dev.humanbrainproject.eu/'
MATOMO_ID_PROD: '12' MATOMO_ID_PROD: '12'
MATOMO_URL_PROD: 'https://stats.humanbrainproject.eu/' MATOMO_URL_PROD: 'https://stats.humanbrainproject.eu/'
PRODUCTION: 'true' PRODUCTION: 'true'
DOCKER_REGISTRY: 'docker-registry.ebrains.eu/siibra/' DOCKER_REGISTRY: 'docker-registry.ebrains.eu/siibra/'
SIIBRA_API_STABLE: 'https://siibra-api-stable.apps.hbp.eu/v1_0' SIIBRA_API_STABLE: 'https://siibra-api-stable.apps.hbp.eu/v1_0'
SIIBRA_API_RC: 'https://siibra-api-rc.apps.hbp.eu/v1_0' SIIBRA_API_RC: 'https://siibra-api-rc.apps.hbp.eu/v1_0'
SIIBRA_API_LATEST: 'https://siibra-api-latest.apps-dev.hbp.eu/v1_0' SIIBRA_API_LATEST: 'https://siibra-api-latest.apps-dev.hbp.eu/v1_0'
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
- name: 'Set matomo env var' - name: 'Set matomo env var'
...@@ -57,6 +57,7 @@ jobs: ...@@ -57,6 +57,7 @@ jobs:
VERSION=$(git rev-parse --short HEAD) VERSION=$(git rev-parse --short HEAD)
echo "EXPERIMENTAL_FEATURE_FLAG=true" >> $GITHUB_ENV echo "EXPERIMENTAL_FEATURE_FLAG=true" >> $GITHUB_ENV
fi fi
echo "Setting VERSION: $VERSION"
echo "VERSION=$VERSION" >> $GITHUB_ENV echo "VERSION=$VERSION" >> $GITHUB_ENV
- name: 'Build docker image' - name: 'Build docker image'
run: | run: |
...@@ -88,7 +89,8 @@ jobs: ...@@ -88,7 +89,8 @@ jobs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
env: env:
GITHUB_API_ROOT: https://api.github.com/repos/fzj-inm1-bda/siibra-explorer GITHUB_API_ROOT: https://api.github.com/repos/fzj-inm1-bda/siibra-explorer
OC_TEMPLATE_NAME: 'siibra-explorer-branch-deploy-2'
needs: build-docker-img needs: build-docker-img
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
...@@ -112,13 +114,13 @@ jobs: ...@@ -112,13 +114,13 @@ jobs:
echo "OKD_URL=https://okd.hbp.eu:443" >> $GITHUB_ENV echo "OKD_URL=https://okd.hbp.eu:443" >> $GITHUB_ENV
echo "OKD_SECRET=${{ secrets.OKD_PROD_SECRET }}" >> $GITHUB_ENV echo "OKD_SECRET=${{ secrets.OKD_PROD_SECRET }}" >> $GITHUB_ENV
echo "OKD_PROJECT=interactive-viewer" >> $GITHUB_ENV echo "OKD_PROJECT=interactive-viewer" >> $GITHUB_ENV
echo "PATH_POSTFIX=" >> $GITHUB_ENV echo "ROUTE_HOST=siibra-explorer.apps.hbp.eu" >> $GITHUB_ENV
echo "Deploy on prod cluster..." echo "Deploy on prod cluster..."
else else
echo "OKD_URL=https://okd-dev.hbp.eu:443" >> $GITHUB_ENV echo "OKD_URL=https://okd-dev.hbp.eu:443" >> $GITHUB_ENV
echo "OKD_SECRET=${{ secrets.OKD_DEV_SECRET }}" >> $GITHUB_ENV echo "OKD_SECRET=${{ secrets.OKD_DEV_SECRET }}" >> $GITHUB_ENV
echo "OKD_PROJECT=interactive-atlas-viewer" >> $GITHUB_ENV echo "OKD_PROJECT=interactive-atlas-viewer" >> $GITHUB_ENV
echo "PATH_POSTFIX=-dev" >> $GITHUB_ENV echo "ROUTE_HOST=siibra-explorer.apps-dev.hbp.eu" >> $GITHUB_ENV
echo "BUILD_TEXT=$BRANCH_NAME" >> $GITHUB_ENV echo "BUILD_TEXT=$BRANCH_NAME" >> $GITHUB_ENV
echo "Deploy on dev cluster..." echo "Deploy on dev cluster..."
fi fi
...@@ -127,39 +129,35 @@ jobs: ...@@ -127,39 +129,35 @@ jobs:
oc login $OKD_URL --token=$OKD_SECRET oc login $OKD_URL --token=$OKD_SECRET
oc project $OKD_PROJECT oc project $OKD_PROJECT
# sanitized branchname == remove _ / and lowercase everything # DEPLOY_ID == remove _ / and lowercase everything from branch
SANITIZED_BRANCH_NAME=$(echo ${BRANCH_NAME//[_\/]/} | awk '{ print tolower($0) }') DEPLOY_ID=$(echo ${BRANCH_NAME//[_\/]/} | awk '{ print tolower($0) }')
echo "SANITIZED_BRANCH_NAME=$SANITIZED_BRANCH_NAME" >> $GITHUB_ENV echo "DEPLOY_ID=$DEPLOY_ID" >> $GITHUB_ENV
echo "Working branch name: $BRANCH_NAME, sanitized branch name: $SANITIZED_BRANCH_NAME"
ROUTE_PATH=/$DEPLOY_ID
echo "ROUTE_PATH=$ROUTE_PATH" >> $GITHUB_ENV
echo "Working branch name: $BRANCH_NAME, deploy_id: $DEPLOY_ID"
# check if the deploy already exist # check if the deploy already exist
if oc get dc siibra-explorer-branch-deploy-$SANITIZED_BRANCH_NAME; then if oc get dc ${{ env.OC_TEMPLATE_NAME }}-$DEPLOY_ID; then
# trigger redeploy if deployconfig exists already # trigger redeploy if deployconfig exists already
echo "dc siibra-explorer-branch-deploy-$SANITIZED_BRANCH_NAME already exist, redeploy..." echo "dc ${{ env.OC_TEMPLATE_NAME }}-$DEPLOY_ID already exist, redeploy..."
oc rollout latest dc/siibra-explorer-branch-deploy-$SANITIZED_BRANCH_NAME oc rollout latest dc/${{ env.OC_TEMPLATE_NAME }}-$DEPLOY_ID
else else
# create new app if deployconfig does not yet exist # create new app if deployconfig does not yet exist
echo "dc siibra-explorer-branch-deploy-$SANITIZED_BRANCH_NAME does not yet exist, create new app..." echo "dc ${{ env.OC_TEMPLATE_NAME }}-$DEPLOY_ID does not yet exist, create new app..."
oc new-app --template siibra-explorer-branch-deploy \ oc new-app --template ${{ env.OC_TEMPLATE_NAME }} \
-p BRANCH_NAME=$BRANCH_NAME \ -p BRANCH_NAME=$BRANCH_NAME \
-p SANITIZED_BRANCH_NAME=$SANITIZED_BRANCH_NAME \ -p DEPLOY_ID=$DEPLOY_ID \
-p PATH_POSTFIX=$PATH_POSTFIX \ -p ROUTE_HOST=$ROUTE_HOST \
-p ROUTE_PATH=$ROUTE_PATH \
-p BUILD_TEXT=$BUILD_TEXT -p BUILD_TEXT=$BUILD_TEXT
fi fi
- name: 'Update status badge' - name: 'Update status badge'
if: success() if: success()
run: | run: |
if [[ "$GITHUB_REF" == 'refs/heads/master' ]] DEPLOY_URL=https://$ROUTE_HOST$ROUTE_PATH
then
DEPLOY_URL="https://siibra-explorer.apps.hbp.eu/master"
elif [[ "$GITHUB_REF" == 'refs/heads/staging' ]]
then
DEPLOY_URL="https://siibra-explorer.apps.hbp.eu/staging"
else
DEPLOY_URL="https://siibra-explorer.apps-dev.hbp.eu/${{ env.SANITIZED_BRANCH_NAME }}"
fi
curl -v \ curl -v \
-X POST \ -X POST \
-H "authorization: Bearer ${{ secrets.GITHUB_TOKEN }}" \ -H "authorization: Bearer ${{ secrets.GITHUB_TOKEN }}" \
...@@ -168,5 +166,7 @@ jobs: ...@@ -168,5 +166,7 @@ jobs:
-d '{ -d '{
"target_url":"'$DEPLOY_URL'", "target_url":"'$DEPLOY_URL'",
"name": "Deployed at OKD", "name": "Deployed at OKD",
"description": "Deployed at OKD",
"context": "[ebrains-okd-deploy] Deployed at OKD",
"state": "success" "state": "success"
}' }'
...@@ -143,7 +143,6 @@ Per [deployment template](./okd-branch-tmpl.yaml), a number of parameters may be ...@@ -143,7 +143,6 @@ Per [deployment template](./okd-branch-tmpl.yaml), a number of parameters may be
| name | required | desc | | name | required | desc |
| --- | --- | --- | | --- | --- | --- |
| `SESSION_SECRET` | | Random strings to encrypt sessions. Not currently used. | | `SESSION_SECRET` | | Random strings to encrypt sessions. Not currently used. |
| `DOCKER_IMAGE_TAG` | true | Dictates which image tag to pull. Currently, possible values are `{latest\|rc\|stable}`. |
| `BRANCH_NAME` | true | Determines the tag of the image to pull. | | `BRANCH_NAME` | true | Determines the tag of the image to pull. |
| `SANITIZED_BRANCH_NAME` | true | Strip all special characters from `BRANCH_NAME`. Acts similar to deploy ID. Distinguishes one deployment from another. Also affects routes: `siibra-explorer-{SANITIZED_BRANCH_NAME}.apps{PATH_POSTFIX}.hbp.eu` | | `SANITIZED_BRANCH_NAME` | true | Strip all special characters from `BRANCH_NAME`. Acts similar to deploy ID. Distinguishes one deployment from another. Also affects routes: `siibra-explorer-{SANITIZED_BRANCH_NAME}.apps{PATH_POSTFIX}.hbp.eu` |
| `PATH_POSTFIX` | | Dictates if postfix, if any, should be added to the route: `siibra-explorer-{DEPLOY_FLAVOUR}.apps{PATH_POSTFIX}.hbp.eu`. Defaults to `''` (empty string). Possible value: `-dev`| | `PATH_POSTFIX` | | Dictates if postfix, if any, should be added to the route: `siibra-explorer-{DEPLOY_FLAVOUR}.apps{PATH_POSTFIX}.hbp.eu`. Defaults to `''` (empty string). Possible value: `-dev`|
......
# Deployment documentation
This document outlines the deployment of `siibra-explorer` on EBRAINS infrastructure.
## Overview
`siibra-explorer` are continuously deployed on openshift container platform hosted by EBRAINS.
The continuous deployment of `siibra-explorer` involves the following steps:
- building docker image
- tag the image, and push to registry
- pull and run the newly built image
## Build
This section outlines the procedure of continuously building and archiving docker images of `siibra-explorer`.
### Images
Docker images are built with [`Dockerfile`](../Dockerfile) by github action withs [yml spec](../.github/workflows/docker_img.yml), and pushed to EBRAINS docker image registry at `docker-registry.ebrains.eu`
`docker-registry.ebrains.eu` is set as the registry
`siibra` is set as the namespace
`siibra-explorer` is set as the image name
The built image will be tagged with the branch name. e.g.
`docker-registry.ebrains.eu/siibra/siibra-explorer:{BRANCH_NAME}`
### Registry
The built docker image will then be pushed to `docker-registry.ebrains.eu` with the access token of a bot account with the rights to push image in `siibra` namespace.
The login credentials are stored in github action secrets:
- username: `{{ secrets.EBRAINS_DOCKER_REG_USER }}`
- access token: `{{ secrets.EBRAINS_DOCKER_REG_TOKEN }}`
> :warning: There are currently no mechanism to delete artefacts from `docker-registry.ebrains.eu`. One must periodically, manually delete untagged images to avoid filling of allotted diskspace.
---
// TODO setup retention policy to allow automatic deletion of artefacts
---
## Deployment
This section outlines how the built image are deployed.
> :information_source: Previous internal guides described a combination of s2i with docker build strategy. This has been demonstrated to be both slow (at build time) and unreliable (over the deployment lifetime).
### Variables
| cluster | name | value |
| --- | --- | --- |
| prod | `PROJECT_NAME` | `interactive-viewer` |
| | `OKD_ENDPOINT` | `https://okd.hbp.eu:443` |
| | `OKD_SECRET` | `{{ secrets.OKD_PROD_SECRET }}` (generated once[1] stored in github action secrets) |
| dev | `PROJECT_NAME` | `interactive-viewer` |
| | `OKD_ENDPOINT` | `https://okd-dev.hbp.eu:443` |
| | `OKD_SECRET` | `{{ secrets.OKD_DEV_SECRET }}` (generated once[1], stored in github action secrets) |
### Triggering deployment
Deployments resides in [docker-img.yml](../.github/workflows/docker_img.yml), `job['trigger-deploy']`. The steps are summarised as below:
- determine if targeting prod or dev cluster.
- If the trigger is update of `master` or `staging` branch, target prod cluster
- Anyother branch, target dev cluster
- login to openshift container platform with the command
```bash
oc login ${OKD_ENDPOINT} --token ${OKD_SECRET}
```
- checkout project with the command
```bash
oc project ${PROJECT_NAME}
```
- check if deployment with name `siibra-explorer-branch-deploy-${SANITIZED_BRANCH_NAME}` exists
- if exists, rollout latest deployment with the command
```bash
oc rollout latest dc/siibra-explorer-branch-deploy-${SANITIZED_BRANCH_NAME}
```
- if does not exist, create new deployment with name `siibra-explorer-branch-deploy-${SANITIZED_BRANCH_NAME}`, using deployment template[2] with corresponding parameters[3]
### [1] OKD service accounts
In order to deploy on OKD clusters in CI/CD pipeline, it is ideal to create a service account. Openshift container platform maintains [a comprehensive guide](https://docs.openshift.com/container-platform/3.11/dev_guide/service_accounts.html) on service account. This section provides a step by step guide on creating the service account.
> :information_source: Why not just use personal access token? 1/ it expires, 2/ it is invalidated when you logout, 3/ (to a less degree, since personal access token has a expiration), revoking personal access token has a greater impact on developer experience, and potentially breaks more things, if one reuses the same personally access token.
#### Prereq
- openshift cli installed (check via `which oc`)
- login command (easiest method to obtain login command: login via web portal > portrait username (*top right*) > Copy Login Command )
#### Configure a Service Account
- login via terminal (paste login command from prereq)
- select the desired project via `oc project ${PROJECT_NAME}`
- create a new SA via: `oc create sa ${SERVICE_ACCOUNT_NAME}`
- get a new token via: `oc sa get-token ${SERVICE_ACCOUNT_NAME}` (store this token securely, ideally in a password manager)
- grant the SA ability to create deployments via: `oc policy add-role-to-user edit -z ${SERVICE_ACCOUNT_NAME}`
### [2] Deployment template
An [openshift template](./okd-branch-tmpl.yaml) has been added to both production (https://okd.hbp.eu) and develop (https://okd-dev.hbp.eu) clusters.
This is done ahead of any deploys, is valid for all future deploys and rarely needs to be updated.
> :warning: The process of editing template is fragile and error prone. One should be vigilant and update the template as little as possible. Ideally, add new templates and alter the deployment pipeline, rather than edit existing templates.
The template is produced mainly by referencing Openshift container platform [template API](https://docs.openshift.com/container-platform/3.11/rest_api/template_openshift_io/template-template-openshift-io-v1.html).
A number of sensitive variables are stored on the openshift clusters, and added to the container at runtime. They include:
| variable name | from | description |
| --- | --- | --- |
| `REDIS_PASSWORD` | `okd_secret.redis-rate-limiting-db-ephemeral.database-password` | password to redis |
| `*` | `okd_configmap.hbp-oauth-config-map.*` | Contains Client ID, Client secret etc for oauth with EBRAINS IAM service |
| `*` | `okd_configmap.fluent-logging.*` | Contains fluentd logging variables |
| `*` | `okd_configmap.plugins.*` | Contains plugins variables |
| `*` | `okd_configmap.other-deploy-config.*` | Contains other deploy variables |
### [3] Deployment parameters
Per [deployment template](./okd-branch-tmpl.yaml), a number of parameters may be required when creating new deployments.
| name | required | desc |
| --- | --- | --- |
| `SESSION_SECRET` | | Random strings to encrypt sessions. Not currently used. |
| `BRANCH_NAME` | true | Determines the tag of the image to pull. |
| `ROUTE_HOST` | true | host field of route service of the deployed template. Possible values are `*.apps.hbp.eu` for prod, `*.apps-dev.hbp.eu` for dev. Must **not** end with `/` |
| `ROUTE_PATH` | | path field of route service of the deployed template. If set, **must** start with `/` |
| `DEPLOY_ID` | true | Strip all special characters from `BRANCH_NAME`. Distinguishes one deployment from another. |
| `BUILD_TEXT` | | Shows as over lay text, to mark dev build. Defaults to `dev build` |
\ No newline at end of file
apiVersion: v1
kind: Template
metadata:
name: siibra-explorer-branch-deploy-2
annotations:
description: "Deploy branch of siibra-explorer"
tags: "nodejs,siibra-explorer"
objects:
- apiVersion: v1
kind: DeploymentConfig
metadata:
name: siibra-explorer-branch-deploy-2-${DEPLOY_ID}
labels:
app: siibra-explorer-branch-deploy-2-${DEPLOY_ID}
spec:
replicas: 3
revisionHistoryLimit: 10
selector:
deploymentconfig: siibra-explorer-branch-deploy-2-${DEPLOY_ID}
template:
metadata:
labels:
app: siibra-explorer-branch-deploy
deploymentconfig: siibra-explorer-branch-deploy-2-${DEPLOY_ID}
spec:
containers:
- env:
- name: SESSION_SECRET
value: ${SESSION_SECRET}
- name: HOSTNAME
value: ${ROUTE_HOST}
- name: HOST_PATHNAME
value: ${ROUTE_PATH}
- name: BUILD_TEXT
value: ${BUILD_TEXT}
- name: REDIS_PASSWORD
valueFrom:
secretKeyRef:
key: database-password
name: redis-rate-limiting-db-ephemeral
envFrom:
- configMapRef:
name: hbp-oauth-config-map
- configMapRef:
name: fluent-logging
- configMapRef:
name: plugins
- configMapRef:
name: other-deploy-config
image: "docker-registry.ebrains.eu/siibra/siibra-explorer:${BRANCH_NAME}"
imagePullPolicy: Always
livenessProbe:
failureThreshold: 3
httpGet:
path: ${ROUTE_PATH}/ready
port: 8080
scheme: HTTP
initialDelaySeconds: 10
periodSeconds: 10
successThreshold: 1
timeoutSeconds: 1
readinessProbe:
failureThreshold: 3
httpGet:
path: ${ROUTE_PATH}/ready
port: 8080
scheme: HTTP
initialDelaySeconds: 3
periodSeconds: 10
successThreshold: 1
timeoutSeconds: 6
name: siibra-explorer-2-${DEPLOY_ID}
ports:
- containerPort: 8080
protocol: TCP
resources: {}
terminationMessagePath: /dev/termination-log
terminationMessagePolicy: File
dnsPolicy: ClusterFirst
restartPolicy: Always
schedulerName: default-scheduler
securityContext: {}
terminationGracePeriodSeconds: 30
- apiVersion: v1
kind: Service
metadata:
labels:
app: siibra-explorer-branch-deploy-2-${DEPLOY_ID}
name: siibra-explorer-branch-deploy-2-${DEPLOY_ID}
spec:
ports:
- name: 8080-tcp
port: 8080
protocol: TCP
targetPort: 8080
selector:
deploymentconfig: siibra-explorer-branch-deploy-2-${DEPLOY_ID}
type: ClusterIP
- apiVersion: v1
kind: Route
metadata:
labels:
app: siibra-explorer-branch-deploy-2-${DEPLOY_ID}
name: siibra-explorer-branch-deploy-2-${DEPLOY_ID}
spec:
host: ${ROUTE_HOST}
path: ${ROUTE_PATH}
port:
targetPort: 8080-tcp
tls:
insecureEdgeTerminationPolicy: Redirect
termination: edge
to:
kind: Service
name: siibra-explorer-branch-deploy-2-${DEPLOY_ID}
weight: 100
wildcardPolicy: None
parameters:
- description: Session secret
from: '[A-Z0-9]{16}'
generate: expression
name: SESSION_SECRET
- name: BRANCH_NAME
required: true
- name: DEPLOY_ID
required: true
description: |
ID that distinguish deployments.
Use only [a-z0-9]{4,}
- name: ROUTE_HOST
description: route/host for the deployed service. Must be unique, or route may not be deployed. Must **NOT** end with with /.
required: true
- name: ROUTE_PATH
description: path for the deployed service. May be left empty. If set, must start with /.
value: ''
- name: BUILD_TEXT
description: 'UI displaying which build'
value: 'dev build'
labels:
template: siibra-explorer-branch-deploy-template-v2
import { getIdFromFullId, strToRgb } from './util' import { getIdFromFullId, strToRgb, verifyPositionArg } from './util'
describe('common/util.js', () => { describe('common/util.js', () => {
describe('getIdFromFullId', () => { describe('getIdFromFullId', () => {
...@@ -76,4 +76,45 @@ describe('common/util.js', () => { ...@@ -76,4 +76,45 @@ describe('common/util.js', () => {
}) })
}) })
describe('verifyPositionArg', () => {
describe('malformed input', () => {
let input
it('> if props.components[0] is string', () => {
input= 'hello world'
expect(verifyPositionArg(input)).toBeFalsy()
})
it('> if position property is object', () => {
input={
x: 0,
y: 0,
z: 0
}
expect(verifyPositionArg(input)).toBeFalsy()
})
it('> if position property is array of incorrect length', () => {
input=[]
expect(verifyPositionArg(input)).toBeFalsy()
})
it('> if position property is array contain non number elements', () => {
input = [1, 2, 'hello world']
expect(verifyPositionArg(input)).toBeFalsy()
})
it('> if position property is array contain NaN', () => {
input=[1,2,NaN]
expect(verifyPositionArg(input)).toBeFalsy()
})
})
describe('correct input', () => {
let input
it('> return true', () => {
expect(verifyPositionArg([1,2,3])).toBeTruthy()
})
})
})
}) })
...@@ -7,9 +7,10 @@ if (process.env.NODE_ENV !== 'production') { ...@@ -7,9 +7,10 @@ if (process.env.NODE_ENV !== 'production') {
if (process.env.FLUENT_HOST) { if (process.env.FLUENT_HOST) {
const Logger = require('./logging') const Logger = require('./logging')
const os = require('os')
const name = process.env.IAV_NAME || 'IAV' const name = process.env.IAV_NAME || 'IAV'
const stage = process.env.IAV_STAGE || 'unnamed-stage' const stage = os.hostname() || 'unknown-host'
const protocol = process.env.FLUENT_PROTOCOL || 'http' const protocol = process.env.FLUENT_PROTOCOL || 'http'
const host = process.env.FLUENT_HOST || 'localhost' const host = process.env.FLUENT_HOST || 'localhost'
......
...@@ -47,7 +47,6 @@ ...@@ -47,7 +47,6 @@
| `FLUENT_HOST` | host for fluent logging | `localhost` | | `FLUENT_HOST` | host for fluent logging | `localhost` |
| `FLUENT_PORT` | port for fluent logging | 24224 | | `FLUENT_PORT` | port for fluent logging | 24224 |
| `IAV_NAME` | application name to be logged | `IAV` | | `IAV_NAME` | application name to be logged | `IAV` |
| `IAV_STAGE` | deploy of the application | `unnamed-stage` |
##### CSP ##### CSP
......
...@@ -3,3 +3,6 @@ ...@@ -3,3 +3,6 @@
## Under the hood ## Under the hood
- Added version check for siibra-api - Added version check for siibra-api
- Updated deploy template
- Remove unused components
- Clean up siibra-api types
# Staging Checklist
**use incognito browser**
[home page](https://siibra-explorer.apps.hbp.eu/staging/)
## General
- [ ] Can access front page
- [ ] Can login to oidc v2 via top-right
## Atlas data specific
- [ ] Human multilevel atlas
- [ ] on click from home page, MNI152, Julich v2.9 loads without issue
- [ ] on hover, show correct region name(s)
- [ ] regional is fine :: select hOC1 right
- [ ] probabilistic map loads fine
- [ ] segmentation layer hides
- [ ] `navigate to` button exists, and works
- [ ] `Open in KG` button exists and works
- [ ] `Description` tabs exists and works
- [ ] `Regional features` tab exists and works
- [ ] `Receptor density` dataset exists and works
- [ ] `Open in KG` button exists and works
- [ ] `Preview` tab exists and works
- [ ] fingerprint is shown, interactable
- [ ] profiles can be loaded, interactable
- [ ] AR can be loaded
- [ ] `IEEG recordings` exists and works (at least 3)
- [ ] GDPR warning triangle
- [ ] `Open in KG` button exists and works
- [ ] perspective view works
- [ ] mesh becomes transparent
- [ ] mesh transparency returns when exit the panel
- [ ] electrodes appear in perspective view
- [ ] some contact points should apepar red (intersect with region)
- [ ] electrode tab
- [ ] show should a number of contact points
- [ ] clicking on electrode should navigate to the contact point location
- [ ] `Connectivity` tab exists and works
- [ ] on opening tab, PMap disappear, colour mapped segmentation appears
- [ ] on closing tab, PMap reappear, segmentation hides
- [ ] Explore in other templates exists, and has MNI152 and big brain
- [ ] clicking on the respective space will load julich 2.9 in that space
- [ ] the navigation should be preserved
- [ ] in big brain v2.9 (or latest)
- [ ] high res hoc1, hoc2, hoc3, lam1-6 are visible
- [ ] Waxholm
- [ ] v4 are visible
- [ ] on hover, show correct region name(s)
\ No newline at end of file
...@@ -58,16 +58,6 @@ input[type="text"] ...@@ -58,16 +58,6 @@ input[type="text"]
height:4rem; height:4rem;
} }
.horizontal-mode .cdk-viewport-wrapper region-list-simple-view
{
width: 200px;
}
.cdk-viewport-wrapper region-list-simple-view
{
width: 100%;
}
.cdk-virtual-scroll-viewport-container.horizontal-mode .cdk-virtual-scroll-viewport-container.horizontal-mode
{ {
height: 6rem; height: 6rem;
......
...@@ -2,9 +2,7 @@ export { ...@@ -2,9 +2,7 @@ export {
ParcellationRegionModule, ParcellationRegionModule,
} from "./module" } from "./module"
export { RegionDirective } from "./region.directive"; export { RegionDirective } from "./region.directive";
export { RegionListSimpleViewComponent } from "./regionListSimpleView/regionListSimpleView.component";
export { RegionMenuComponent } from "./regionMenu/regionMenu.component"; export { RegionMenuComponent } from "./regionMenu/regionMenu.component";
export { SimpleRegionComponent } from "./regionSimple/regionSimple.component"; export { SimpleRegionComponent } from "./regionSimple/regionSimple.component";
export { RenderViewOriginDatasetLabelPipe } from "./region.base"; export { RenderViewOriginDatasetLabelPipe } from "./region.base";
\ No newline at end of file
...@@ -5,7 +5,6 @@ import { AngularMaterialModule } from "src/sharedModules"; ...@@ -5,7 +5,6 @@ import { AngularMaterialModule } from "src/sharedModules";
import { UtilModule } from "src/util"; import { UtilModule } from "src/util";
import { RenderViewOriginDatasetLabelPipe } from "./region.base"; import { RenderViewOriginDatasetLabelPipe } from "./region.base";
import { RegionDirective } from "./region.directive"; import { RegionDirective } from "./region.directive";
import { RegionListSimpleViewComponent } from "./regionListSimpleView/regionListSimpleView.component";
import { RegionMenuComponent } from "./regionMenu/regionMenu.component"; import { RegionMenuComponent } from "./regionMenu/regionMenu.component";
import { SimpleRegionComponent } from "./regionSimple/regionSimple.component"; import { SimpleRegionComponent } from "./regionSimple/regionSimple.component";
import { BSFeatureModule } from "../regionalFeatures/bsFeatures"; import { BSFeatureModule } from "../regionalFeatures/bsFeatures";
...@@ -25,7 +24,6 @@ import { HttpClientModule } from "@angular/common/http"; ...@@ -25,7 +24,6 @@ import { HttpClientModule } from "@angular/common/http";
], ],
declarations: [ declarations: [
RegionMenuComponent, RegionMenuComponent,
RegionListSimpleViewComponent,
SimpleRegionComponent, SimpleRegionComponent,
RegionDirective, RegionDirective,
...@@ -34,7 +32,6 @@ import { HttpClientModule } from "@angular/common/http"; ...@@ -34,7 +32,6 @@ import { HttpClientModule } from "@angular/common/http";
], ],
exports: [ exports: [
RegionMenuComponent, RegionMenuComponent,
RegionListSimpleViewComponent,
SimpleRegionComponent, SimpleRegionComponent,
RegionDirective, RegionDirective,
......
...@@ -471,53 +471,51 @@ describe('> region.base.ts', () => { ...@@ -471,53 +471,51 @@ describe('> region.base.ts', () => {
it('> does not populate if position property is absent', () => { it('> does not populate if position property is absent', () => {
regionBase.region = { regionBase.region = {
...mr0 ...mr0
} } as any
expect(regionBase.position).toBeFalsy() expect(regionBase.position).toBeFalsy()
}) })
describe('> does not populate if position property is malformed', () => { describe('> does not populate if position property is malformed', () => {
it('> if region is falsy', () => {
it('> if position property is string', () => { regionBase.region = null
regionBase.region = {
...mr0,
position: 'hello world'
}
expect(regionBase.position).toBeFalsy() expect(regionBase.position).toBeFalsy()
}) })
it('> if position property is object', () => { it('> if props is falsy', () => {
regionBase.region = { regionBase.region = {
...mr0, ...mr0,
position: { props: null
x: 0, } as any
y: 0,
z: 0
}
}
expect(regionBase.position).toBeFalsy() expect(regionBase.position).toBeFalsy()
}) })
it('> if props.components is falsy', () => {
it('> if position property is array of incorrect length', () => {
regionBase.region = { regionBase.region = {
...mr0, ...mr0,
position: [] props: {
} components: null
}
} as any
expect(regionBase.position).toBeFalsy() expect(regionBase.position).toBeFalsy()
}) })
it('> if props.components[0] is falsy', () => {
it('> if position property is array contain non number elements', () => {
regionBase.region = { regionBase.region = {
...mr0, ...mr0,
position: [1, 2, 'hello world'] props: {
} components: []
}
} as any
expect(regionBase.position).toBeFalsy() expect(regionBase.position).toBeFalsy()
}) })
it('> if props.components[0].centroid is falsy', () => {
it('> if position property is array contain NaN', () => {
regionBase.region = { regionBase.region = {
...mr0, ...mr0,
position: [1, 2, NaN] props: {
} components: [{
centroid: null
}]
}
} as any
expect(regionBase.position).toBeFalsy() expect(regionBase.position).toBeFalsy()
}) })
}) })
...@@ -525,8 +523,12 @@ describe('> region.base.ts', () => { ...@@ -525,8 +523,12 @@ describe('> region.base.ts', () => {
it('> populates if position property is array with length 3 and non NaN element', () => { it('> populates if position property is array with length 3 and non NaN element', () => {
regionBase.region = { regionBase.region = {
...mr0, ...mr0,
position: [1, 2, 3] props: {
} components: [{
centroid: [1, 2, 3]
}]
},
} as any
expect(regionBase.position).toBeTruthy() expect(regionBase.position).toBeTruthy()
}) })
}) })
...@@ -549,7 +551,7 @@ describe('> region.base.ts', () => { ...@@ -549,7 +551,7 @@ describe('> region.base.ts', () => {
const regionBase = new RegionBase(mockStore) const regionBase = new RegionBase(mockStore)
regionBase.region = { regionBase.region = {
rgb: [100, 120, 140] rgb: [100, 120, 140]
} } as any
expect( expect(
regionBase.rgbString regionBase.rgbString
).toEqual(`rgb(100,120,140)`) ).toEqual(`rgb(100,120,140)`)
...@@ -560,7 +562,7 @@ describe('> region.base.ts', () => { ...@@ -560,7 +562,7 @@ describe('> region.base.ts', () => {
const regionBase = new RegionBase(mockStore) const regionBase = new RegionBase(mockStore)
regionBase.region = { regionBase.region = {
labelIndex: 65535 labelIndex: 65535
} } as any
expect( expect(
regionBase.rgbString regionBase.rgbString
).toEqual(`rgb(255,255,255)`) ).toEqual(`rgb(255,255,255)`)
...@@ -576,7 +578,7 @@ describe('> region.base.ts', () => { ...@@ -576,7 +578,7 @@ describe('> region.base.ts', () => {
ngId: 'foo', ngId: 'foo',
name: 'bar', name: 'bar',
labelIndex: 152 labelIndex: 152
} } as any
expect(strToRgbSpy).toHaveBeenCalledWith(`foo152`) expect(strToRgbSpy).toHaveBeenCalledWith(`foo152`)
}) })
it('> if ngId is not defined, use name', () => { it('> if ngId is not defined, use name', () => {
...@@ -585,7 +587,7 @@ describe('> region.base.ts', () => { ...@@ -585,7 +587,7 @@ describe('> region.base.ts', () => {
regionBase.region = { regionBase.region = {
name: 'bar', name: 'bar',
labelIndex: 152 labelIndex: 152
} } as any
expect(strToRgbSpy).toHaveBeenCalledWith(`bar152`) expect(strToRgbSpy).toHaveBeenCalledWith(`bar152`)
}) })
}) })
...@@ -601,7 +603,7 @@ describe('> region.base.ts', () => { ...@@ -601,7 +603,7 @@ describe('> region.base.ts', () => {
const regionBase = new RegionBase(mockStore) const regionBase = new RegionBase(mockStore)
regionBase.region = { regionBase.region = {
foo: 'bar' foo: 'bar'
} } as any
expect( expect(
regionBase.rgbString regionBase.rgbString
).toEqual(`rgb(${arr.join(',')})`) ).toEqual(`rgb(${arr.join(',')})`)
...@@ -613,7 +615,7 @@ describe('> region.base.ts', () => { ...@@ -613,7 +615,7 @@ describe('> region.base.ts', () => {
const regionBase = new RegionBase(mockStore) const regionBase = new RegionBase(mockStore)
regionBase.region = { regionBase.region = {
foo: 'bar' foo: 'bar'
} } as any
expect( expect(
regionBase.rgbString regionBase.rgbString
).toEqual(`rgb(255,200,200)`) ).toEqual(`rgb(255,200,200)`)
......
...@@ -8,6 +8,8 @@ import { viewerStateSetConnectivityRegion, viewerStateNavigateToRegion, viewerSt ...@@ -8,6 +8,8 @@ import { viewerStateSetConnectivityRegion, viewerStateNavigateToRegion, viewerSt
import { viewerStateFetchedTemplatesSelector, viewerStateGetSelectedAtlas, viewerStateSelectedTemplateFullInfoSelector, viewerStateSelectedTemplateSelector } from "src/services/state/viewerState/selectors"; import { viewerStateFetchedTemplatesSelector, viewerStateGetSelectedAtlas, viewerStateSelectedTemplateFullInfoSelector, viewerStateSelectedTemplateSelector } from "src/services/state/viewerState/selectors";
import { strToRgb, verifyPositionArg, getRegionHemisphere } from 'common/util' import { strToRgb, verifyPositionArg, getRegionHemisphere } from 'common/util'
import { getPosFromRegion } from "src/util/siibraApiConstants/fn"; import { getPosFromRegion } from "src/util/siibraApiConstants/fn";
import { TRegionDetail } from "src/util/siibraApiConstants/types";
import { IHasId } from "src/util/interfaces";
@Directive() @Directive()
export class RegionBase { export class RegionBase {
...@@ -15,7 +17,14 @@ export class RegionBase { ...@@ -15,7 +17,14 @@ export class RegionBase {
public rgbString: string public rgbString: string
public rgbDarkmode: boolean public rgbDarkmode: boolean
private _region: any private _region: TRegionDetail & {
context?: {
atlas: IHasId
template: IHasId
parcellation: IHasId
}
ngId?: string
}
private _position: [number, number, number] private _position: [number, number, number]
set position(val){ set position(val){
...@@ -34,16 +43,12 @@ export class RegionBase { ...@@ -34,16 +43,12 @@ export class RegionBase {
set region(val) { set region(val) {
this._region = val this._region = val
this.region$.next(this._region) this.region$.next(this._region)
this.hasContext$.next(!!this._region.context) this.hasContext$.next(!!this._region?.context)
this.position = null this.position = null
// bug the centroid returned is currently nonsense // bug the centroid returned is currently nonsense
// this.position = val?.props?.centroid_mm // this.position = val?.props?.centroid_mm
if (!this._region) return if (!this._region) return
if (val?.position) {
this.position = val?.position
}
const pos = getPosFromRegion(val) const pos = getPosFromRegion(val)
if (pos) { if (pos) {
this.position = pos this.position = pos
...@@ -65,7 +70,7 @@ export class RegionBase { ...@@ -65,7 +70,7 @@ export class RegionBase {
get originDatainfos(){ get originDatainfos(){
if (!this._region) return [] if (!this._region) return []
return this._region._dataset_specs || this._region.originDatainfos return (this._region._dataset_specs || []).filter(spec => spec['@type'] === 'minds/core/dataset/v1.0.0')
} }
public hasContext$: BehaviorSubject<boolean> = new BehaviorSubject(false) public hasContext$: BehaviorSubject<boolean> = new BehaviorSubject(false)
......
import { Component, Input } from "@angular/core";
import { Store } from "@ngrx/store";
import { IavRootStoreInterface } from "src/services/stateStore.service";
import { RegionBase } from '../region.base'
@Component({
selector: 'region-list-simple-view',
templateUrl: './regionListSimpleView.template.html',
styleUrls: [
'./regionListSimpleView.style.css',
],
})
export class RegionListSimpleViewComponent extends RegionBase {
@Input()
public showBrainIcon: boolean = false
@Input()
public showDesc: boolean = false
constructor(
store$: Store<IavRootStoreInterface>,
) {
super(store$)
}
}
<!-- selected brain region -->
<div class="flex-grow-1 flex-shrink-1 pt-2 pb-2 d-flex flex-row align-items-center flex-nowrap">
<i *ngIf="showBrainIcon" class="flex-grow-0 flex-shrink-0 fas fa-brain mr-2"></i>
<small class="flex-grow-1 flex-shrink-1 ">
{{ region.name }}
</small>
<button mat-icon-button
*ngIf="position"
class="flex-grow-0 flex-shrink-0"
(click)="navigateToRegion()" >
<i *ngIf="isSelected" class="fas fa-map-marked-alt"></i>
</button>
<button mat-icon-button
class="flex-grow-0 flex-shrink-0"
(click)="toggleRegionSelected()" >
<i *ngIf="isSelected" class="fas fa-trash"></i>
<i *ngIf="!isSelected" class="fas fa-plus"></i>
</button>
</div>
<mat-divider *ngIf="showDesc && region.description"></mat-divider>
<small *ngIf="showDesc && region.description">
{{ region.description }}
</small>
\ No newline at end of file
...@@ -10,7 +10,6 @@ ...@@ -10,7 +10,6 @@
<mat-card-title> <mat-card-title>
<div class="position-relative region-name iv-custom-comp text"> <div class="position-relative region-name iv-custom-comp text">
{{ region.name }} {{ region.name }}
<small *ngIf="region.status"> ({{region.status}})</small>
</div> </div>
</mat-card-title> </mat-card-title>
...@@ -22,20 +21,7 @@ ...@@ -22,20 +21,7 @@
</span> </span>
<!-- origin datas format --> <!-- origin datas format -->
<div *ngFor="let originDataset of (region.originDatasets || []); let index = index"
class="ml-2">
<i>&#183;</i>
<span *ngIf="originDataset?.format?.name as regionOrDsFormatName; else fallbackODsnameTmpl">
{{ regionOrDsFormatName }}
</span>
<ng-template #fallbackODsnameTmpl>
<span>
{{ regionOriginDatasetLabels$ | async | renderViewOriginDatasetlabel : index }}
</span>
</ng-template>
</div>
<mat-divider vertical="true" class="ml-2 h-2rem"></mat-divider> <mat-divider vertical="true" class="ml-2 h-2rem"></mat-divider>
<!-- position --> <!-- position -->
......
<div class="d-flex flex-row"> <div class="d-flex flex-row">
<small class="text-truncate flex-shrink-1 flex-grow-1"> <small class="text-truncate flex-shrink-1 flex-grow-1">
{{ region.name }}<span *ngIf="region.status" class="text-muted">{{' (' + region.status + ')'}}</span> {{ region.name }}
</small> </small>
<div class="flex-grow-0 flex-shrink-0 d-flex flex-row"> <div class="flex-grow-0 flex-shrink-0 d-flex flex-row">
......
0% or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment