From 991d6dbfd9fd9184b31c88704b191c801043c4dc Mon Sep 17 00:00:00 2001 From: Aiosa <469130@mail.muni.cz> Date: Sun, 5 Jan 2025 17:49:35 +0100 Subject: [PATCH 1/6] feat: winter school features + bugfixes: 1 --- CHANGELOG.md | 7 +- modules/annotations/annotations.js | 90 +++++----- modules/annotations/presets.js | 27 ++- modules/empaia-wsi-tile-source/tile-source.js | 6 +- modules/empation-api/empationapi.js | 4 +- plugins/annotations/annotationsGUI.js | 57 ++++-- plugins/annotations/include.json | 2 - plugins/empaia/empaia.js | 168 +++++++++++++----- server/php/init.php | 5 +- src/app.js | 10 +- src/external/scalebar.js | 20 ++- src/loader.js | 7 + src/parse-input.js | 19 +- 13 files changed, 289 insertions(+), 133 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 0335ac12..cba157af 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,7 +4,12 @@ The changelog file describes changes made since v2.0.0, which made significant c to the versions 1.x.x. ### Unreleased 2.1.1 -**Features:** standalone wsi tile source module. +**Features:** standalone wsi tile source module. Edge navigation optional. + +**Bugfixes:** OIDC module popup method - await login. +Use session storage to store xOpat sessions as well. +Fixed scalebar magnification estimates. Annotations IO bugfixes. +Extend await event support. ### 2.1.0 **Features:** new system for module/plugin building, improvements of annotation listing features, diff --git a/modules/annotations/annotations.js b/modules/annotations/annotations.js index 704ab0b7..268e9846 100644 --- a/modules/annotations/annotations.js +++ b/modules/annotations/annotations.js @@ -703,6 +703,13 @@ window.OSDAnnotations = class extends XOpatModuleSingleton { object.zooming(this.canvas.computeGraphicZoom(zoom), zoom); } + setCloseEdgeMouseNavigation(enable) { + window.removeEventListener("mousemove", this._edgesMouseNavigation); + if (enable) { + window.addEventListener("mousemove", this._edgesMouseNavigation); + } + } + /************************ Layers *******************************/ /** @@ -1047,6 +1054,19 @@ window.OSDAnnotations = class extends XOpatModuleSingleton { return o.hasOwnProperty("incrementId") && o.hasOwnProperty("sessionID"); } + /** + * Find annotations by a predicate + * @param callback + * @return {*} + */ + find(callback) { + return this.canvas._objects.find(callback); + } + + filter(callback) { + return this.canvas._objects.filter(callback); + } + /** * Delete object without knowledge of its identity (fully-fledged annotation or helper one) * @param {fabric.Object} o @@ -1540,49 +1560,9 @@ window.OSDAnnotations = class extends XOpatModuleSingleton { _this.mode.handleMouseMove(o.e, screenToPixelCoords(o.e.x, o.e.y)); } else { _this.mode.handleMouseHover(o.e, screenToPixelCoords(o.e.x, o.e.y)); - } }); - // Cached event that keeps moving the viewport is not useful since it keeps moving when user exist the window - // let _lastCalled = 0; - // let _cachedEvent = null; - const mouseNavigation = e => { - // const now = Date.now(); - if (this.mode !== this.Modes.AUTO /*|| now - _lastCalled > 30*/) { - //_cachedEvent = e || _cachedEvent; // keep reference to the most recent event - - const edgeThreshold = 20; - // const mouseX = _cachedEvent.clientX; - // const mouseY = _cachedEvent.clientY; - const mouseX = e.clientX; - const mouseY = e.clientY; - - const nearLeftEdge = mouseX >= 0 && edgeThreshold - mouseX; - const nearTopEdge = mouseY >= 0 && edgeThreshold / 2 - mouseY; //top edge near - const nearRightEdge = mouseX - window.innerWidth + edgeThreshold; - const nearBottomEdge = mouseY - window.innerHeight + edgeThreshold; - - if ( - (nearTopEdge < edgeThreshold && nearTopEdge > 0) || - (nearRightEdge < edgeThreshold && nearRightEdge > 0) || - (nearBottomEdge < edgeThreshold && nearBottomEdge > 0) || - (nearLeftEdge < edgeThreshold && nearLeftEdge > 0) - ) { - const center = VIEWER.viewport.getCenter(true); - //const current = VIEWER.viewport.windowToViewportCoordinates(new OpenSeadragon.Point(_cachedEvent.x, _cachedEvent.y)); - const current = VIEWER.viewport.windowToViewportCoordinates(new OpenSeadragon.Point(e.x, e.y)); - let direction = current.minus(center); - direction = direction.divide(Math.sqrt(Math.pow(direction.x, 2) + Math.pow(direction.y, 2))); - VIEWER.viewport.panTo(direction.times(0.004 / VIEWER.scalebar.imagePixelSizeOnScreen()).plus(center)); - //_lastCalled = now; - //setTimeout(mouseNavigation, 35); - } - } - }; - - window.addEventListener("mousemove", mouseNavigation); - this.canvas.on('mouse:wheel', function (o) { if (_this.disabledInteraction) return; @@ -1618,6 +1598,9 @@ window.OSDAnnotations = class extends XOpatModuleSingleton { // Wheel while viewer runs not enabled because this already performs zoom. // VIEWER.addHandler("canvas-scroll", function (e) { ... }); + + // Rewrite with bind this arg to use in events + this._edgesMouseNavigation = this._edgesMouseNavigation.bind(this); } static _registerAnnotationFactory(FactoryClass, atRuntime) { @@ -1798,6 +1781,33 @@ window.OSDAnnotations = class extends XOpatModuleSingleton { }); //todo rethrow? rewrite as async call with try finally } + _edgesMouseNavigation(e) { + if (this.mode !== this.Modes.AUTO) { + + const edgeThreshold = 20; + const mouseX = e.clientX; + const mouseY = e.clientY; + + const nearLeftEdge = mouseX >= 0 && edgeThreshold - mouseX; + const nearTopEdge = mouseY >= 0 && edgeThreshold / 2 - mouseY; //top edge near + const nearRightEdge = mouseX - window.innerWidth + edgeThreshold; + const nearBottomEdge = mouseY - window.innerHeight + edgeThreshold; + + if ( + (nearTopEdge < edgeThreshold && nearTopEdge > 0) || + (nearRightEdge < edgeThreshold && nearRightEdge > 0) || + (nearBottomEdge < edgeThreshold && nearBottomEdge > 0) || + (nearLeftEdge < edgeThreshold && nearLeftEdge > 0) + ) { + const center = VIEWER.viewport.getCenter(true); + const current = VIEWER.viewport.windowToViewportCoordinates(new OpenSeadragon.Point(e.x, e.y)); + let direction = current.minus(center); + direction = direction.divide(Math.sqrt(Math.pow(direction.x, 2) + Math.pow(direction.y, 2))); + VIEWER.viewport.panTo(direction.times(0.004 / VIEWER.scalebar.imagePixelSizeOnScreen()).plus(center)); + } + } + }; + // Copied out of OpenSeadragon private code scope to allow manual scroll navigation _fireMouseWheelNavigation(event) { // Simulate a 'wheel' event diff --git a/modules/annotations/presets.js b/modules/annotations/presets.js index e2728a88..9b730607 100644 --- a/modules/annotations/presets.js +++ b/modules/annotations/presets.js @@ -203,7 +203,7 @@ OSDAnnotations.PresetManager = class { */ isUnusedPreset(p) { return !p._used && p.objectFactory == this._context.polygonFactory - && p.meta.category?.value === "" + && !p.meta.category?.value && Object.keys(p.meta).length === 1; } @@ -416,13 +416,12 @@ OSDAnnotations.PresetManager = class { async import(presets, clear=false) { const _this = this; - if (clear) { - for (let pid in this._presets) { - const preset = this._presets[pid]; - if (this.isUnusedPreset(preset)) { - this._context.raiseEvent('preset-delete', {preset}); - delete this._presets[pid]; - } + for (let pid in this._presets) { + const preset = this._presets[pid]; + // TODO: clear might remove presets that are attached to existing annotations! + if (clear || this.isUnusedPreset(preset)) { + this._context.raiseEvent('preset-delete', {preset}); + delete this._presets[pid]; } } @@ -530,12 +529,12 @@ OSDAnnotations.PresetManager = class { let f = h * 6 - i; let q = 1 - f; switch(i % 6){ - case 0: r = 1; g = f; b = 0; break; - case 1: r = q; g = 1; b = 0; break; - case 2: r = 0; g = 1; b = f; break; - case 3: r = 0; g = q; b = 1; break; - case 4: r = f; g = 0; b = 1; break; - case 5: r = 1; g = 0; b = q; break; + case 0: r = 1; g = f; b = 0; break; + case 1: r = q; g = 1; b = 0; break; + case 2: r = 0; g = 1; b = f; break; + case 3: r = 0; g = q; b = 1; break; + case 4: r = f; g = 0; b = 1; break; + case 5: r = 1; g = 0; b = q; break; } let c = "#" + ("00" + (~ ~(r * 255)).toString(16)).slice(-2) + ("00" + (~ ~(g * 255)).toString(16)).slice(-2) diff --git a/modules/empaia-wsi-tile-source/tile-source.js b/modules/empaia-wsi-tile-source/tile-source.js index a2c0073b..62872495 100644 --- a/modules/empaia-wsi-tile-source/tile-source.js +++ b/modules/empaia-wsi-tile-source/tile-source.js @@ -25,11 +25,11 @@ OpenSeadragon.EmpaiaStandaloneV3TileSource = class extends OpenSeadragon.TileSou */ supports( data, url ) { if (url && Array.isArray(data)) { - //multi-tile or single tile access - let match = url.match(/^(\/?[^\/].*\/v3\/files)\/info/i); + //multi-tile or single tile access, batch is old name on the api + let match = url.match(/^(\/?[^\/].*\/v3\/)(files|batch)\/info/i); if (match) { data = data || [{}]; - data[0].tilesUrl = match[1]; + data[0].tilesUrl = match[1] + "batch"; return true; } } else if (url && typeof data === "object") { diff --git a/modules/empation-api/empationapi.js b/modules/empation-api/empationapi.js index 7b031b3e..1a7049e5 100644 --- a/modules/empation-api/empationapi.js +++ b/modules/empation-api/empationapi.js @@ -1,2 +1,2 @@ -var EmpationAPI;(()=>{"use strict";var t={d:(e,i)=>{for(var n in i)t.o(i,n)&&!t.o(e,n)&&Object.defineProperty(e,n,{enumerable:!0,get:i[n]})},o:(t,e)=>Object.prototype.hasOwnProperty.call(t,e),r:t=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(t,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(t,"__esModule",{value:!0})}},e={};t.r(e),t.d(e,{AbstractAPI:()=>c,EventSource:()=>n,HTTPError:()=>o,Logger:()=>l,RawAPI:()=>a,RootAPI:()=>p,RootContext:()=>f,STATUS_CODES:()=>s,ScopeAPI:()=>y,ScopeContext:()=>v,V3:()=>i,getJwtTokenExpiresTimeout:()=>u,parseJwtToken:()=>d,sleep:()=>h});var i={};t.r(i),t.d(i,{Apps:()=>g,Cases:()=>C,Examinations:()=>$,RationAI:()=>rt,Root:()=>ct,Scope:()=>J,Slides:()=>z,Storage:()=>k});class n{constructor(){this.events={}}addOnceHandler(t,e,i,n,s){const r=this;n=n||1;let o=0;const a=function(i){return o++,o===n&&r.removeHandler(t,a),e(i)};this.addHandler(t,a,i,s)}addHandler(t,e,i=null,s=0){let r=this.events[t];if(r||(this.events[t]=r=[]),e&&n.isFunction(e)){let t=r.length,n={handler:e,userData:i||null,priority:s||0};for(r[t]=n;t>0&&r[t-1].priority{const r=e.length;!function o(a){if(a>=r||!e[a])return s("Resolved!"),null;i.eventSource=t,i.userData=e[a].userData;let c=e[a].handler(i);return c=c&&"promise"===n.type(c)?c:Promise.resolve(),c.then((()=>o(a+1)))}(0)}))}):null}raiseEvent(t,e){const i=this.getHandler(t);if(i)return i(this,e||{})}raiseEventAwaiting(t,e){const i=this.getAwaitingHandler(t);return i?i(this,e||{}):Promise.resolve("No handler for this event registered.")}static isFunction(t){return"function"===this.type(t)}static type(t){return null==t?String(t):this.class2type[t.toString()]||("function"==typeof t?"function":"object")}}n.class2type={"[object Boolean]":"boolean","[object Number]":"number","[object String]":"string","[object Function]":"function","[object AsyncFunction]":"function","[object Promise]":"promise","[object Array]":"array","[object Date]":"date","[object RegExp]":"regexp","[object Object]":"object"};const s={100:"Continue",101:"Switching protocols",102:"Processing",103:"Early Hints",200:"Ok",201:"Created",202:"Accepted",203:"Non Authoritative Information",204:"No Content",205:"Reset Content",206:"Partial Content",207:"Multi Status",300:"Multiple Choices",301:"Moved Permanently",302:"Moved Temporarily",303:"See Other",304:"Not Modified",305:"Use Proxy",307:"Temporary Redirect",308:"Permanent Redirect",400:"Bad Request",401:"Unauthorized",402:"Payment Required",403:"Forbidden",404:"Not Found",405:"Method Not Allowed",406:"Not Acceptable",407:"Proxy Authentication Required",408:"Request Timeout",409:"Conflict",410:"Gone",411:"Length Required",412:"Precondition Failed",413:"Request Too Long",414:"Request Uri Too Long",415:"Unsupported Media Type",416:"Requested Range Not Satisfiable",417:"Expectation Failed",418:"Im A Teapot",419:"Insufficient Space On Resource",420:"Method Failure",421:"Misdirected Request",422:"Unprocessable Entity",423:"Locked",424:"Failed Dependency",426:"Upgrade Required",428:"Precondition Required",429:"Too Many Requests",431:"Request Header Fields Too Large",451:"Unavailable For Legal Reasons",500:"Internal Server Error",501:"Not Implemented",502:"Bad Gateway",503:"Service Unavailable",504:"Gateway Timeout",505:"Http Version Not Supported",507:"Insufficient Storage",511:"Network Authentication Required"};var r=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class o extends Error{constructor(t,e,i){super(e||s[t]||`HTTP Code ${t}`),arguments.length>=3&&i&&Object.assign(this,i),this.name=function(t){const e=4==(t/100|0)||5==(t/100|0)?"error":"";return` ${String(s[t]||`HTTP Code ${t}`).replace(/error$/i,"")} ${e}`.split(" ").reduce(((t,e)=>t+(e?e.charAt(0).toUpperCase()+e.slice(1):"")))}(t),this.statusCode=t}}class a{constructor(t,e={}){this.url=t,this.options=e}_parseQueryParams(t){if(t){if("string"==typeof t)return t;if(t.constructor===Object||void 0===t.constructor)for(let e in t){null==t[e]&&delete t[e]}return`?${new URLSearchParams(t)}`}return""}_fetch(t,e){return r(this,void 0,void 0,(function*(){const i=yield fetch(t,{method:e.method,headers:e.headers,body:e.body});let n;try{n=yield i[e.responseType||"json"]()}catch(e){throw new o(500,`Failed to parse response data. Original status: ${i.status} | ${i.statusText}`,{url:t,error:e})}if(!i.ok)throw new o(i.status,i.statusText,n);return n}))}http(t,e){return r(this,void 0,void 0,(function*(){const i=!!e.body;return e.method=e.method||(i?"POST":"GET"),t.startsWith("/")||(t=`/${t}`),e.query=this._parseQueryParams(e.query),e.headers=e.headers||{},e.headers["Content-Type"]="application/json",e.body&&"string"!=typeof e.body?e.body=JSON.stringify(e.body):e.body=void 0,yield this._fetch(this.url+t+e.query,e)}))}}class c extends n{getCallerName(){const t=Error.prepareStackTrace;Error.prepareStackTrace=(t,e)=>e;const{stack:e}=new Error;Error.prepareStackTrace=t;return(null==e?void 0:e[2])||"unknown context"}requires(t,e){if(!e)throw`ArgumentError[${this.getCallerName()}] ${t} is missing - required property!`}}function u(t){return 1e3*t.exp-Date.now()||3e5}function d(t){return JSON.parse(atob(t.split(".")[1]))}function h(t){return new Promise((e=>setTimeout(e,t)))}class l{static error(...t){console.error("E:EmpationAPI",...t)}static warn(...t){console.warn("W:EmpationAPI",...t)}static info(...t){console.info("I:EmpationAPI",...t)}static debug(...t){console.debug("D:EmpationAPI",...t)}}class f{}class p extends c{constructor(t){if(super(),this.accessToken=null,this._tokenExpires=0,this._rawToken="",!t.workbenchApiUrl)throw"WB Api url is required!";let e;e=t.apiRootPath?t.apiRootPath.startsWith("/")?`${t.workbenchApiUrl}${t.apiRootPath}`:`${t.workbenchApiUrl}/${t.apiRootPath}`:t.workbenchApiUrl,e.endsWith("/")&&(e=e.slice(0,-1)),this.options={apiUrl:e,workbenchApiUrl:t.workbenchApiUrl,anonymousUserId:t.anonymousUserId||"anonymous",apiRootPath:t.apiRootPath||""},this._userId=this.options.anonymousUserId,this.cached={}}from(t,e=!0){if(!t)return this.reset();this._rawToken=t,e=e&&!this.accessToken,this.accessToken=d(t);const i=u(this.accessToken);this._tokenExpires=Date.now()+i/2;let n=this.accessToken.sub;if(!n)throw"Invalid User ID! Must be valid string shorter than 50 characters!";n.length>50&&(console.warn("User ID exceeded 50 characters! Using User ID shortened to first 50 characters!"),n=n.slice(0,50)),this.userId!==n&&(this._userId=n,e&&this.raiseEvent("init"))}use(t,e=!0){if(e=e&&!this._userId,this.reset(),!t||t.length>50)throw"Invalid User ID! Must be valid string shorter than 50 characters!";this._userId=t,e&&this.raiseEvent("init_no_token")}reset(){this._rawToken="",this._tokenExpires=0,this.accessToken=null,this._userId=this.options.anonymousUserId,this.defaultScopeKey="",this.scopes.forEach((t=>t.reset())),this.scopes.clear(),this.raiseEvent("reset")}get userId(){return this._userId}get rawToken(){return this._rawToken}rawQuery(t,e){return i=this,n=void 0,r=function*(){if(!this._userId)throw"User must be configured to access Empaia API: either provide a valid 'anonymous' user ID through env variables, or configure the Root API with a valid token.";if(this._tokenExpires>0&&Date.now()>this._tokenExpires){const t={newToken:""};yield this.raiseEventAwaiting("token-refresh",t),this.from(t.newToken)}},new((s=void 0)||(s=Promise))((function(t,e){function o(t){try{c(r.next(t))}catch(t){e(t)}}function a(t){try{c(r.throw(t))}catch(t){e(t)}}function c(e){var i;e.done?t(e.value):(i=e.value,i instanceof s?i:new s((function(t){t(i)}))).then(o,a)}c((r=r.apply(i,n||[])).next())}));var i,n,s,r}}class v{}class y extends c{}var m=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class g extends f{constructor(t){super(),this.data=null,this._defaultApp=null,this.context=t}list(){return m(this,void 0,void 0,(function*(){return this.data=yield this.context.rawQuery("/apps/query",{method:"PUT",body:{apps:null,tissues:null,stains:null,job_modes:null}})}))}query(t){return m(this,void 0,void 0,(function*(){return this.data=yield this.context.rawQuery("/apps/query",{method:"PUT",body:t})}))}default(){return m(this,void 0,void 0,(function*(){this.data||(yield this.list());for(let t of this.data.items)if("MAP3"===t.name_short&&"rationai"===t.vendor_name){this._defaultApp=t;break}if(!this._defaultApp)throw"Default APP not present in the infrastructure! Was it imported?";return this._defaultApp}))}}const x=t=>("string"==typeof t&&(t=Number(t)),t),w=(t,e,i,n)=>{const s=new RegExp(e).exec(t);return!(!s||i<1||i>=s.length)&&s[i]===n};var b=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class T{constructor(t){this.customCases=null,this.caseHierarchy=null,this.caseTissues=null,this.caseStains=null,this.identifierSeparator="",this.hierarchySpec=[],this.hierarchyNameOverrides={},this.context=t}use(t,e,i={}){this.hierarchySpec=e,this.identifierSeparator=t,this.hierarchyNameOverrides=i}getCustomCases(){return b(this,void 0,void 0,(function*(){return this.customCases||(this.customCases=(yield this.context.list()).items.map((t=>Object.assign(Object.assign({},t),{pathInHierarchy:this.getCaseHierarchyPath(t)})))),this.customCases}))}getCaseHierarchyPath(t){if(!this.identifierSeparator||!this.hierarchySpec)throw"ArgumentError[CaseExplorer] identifierSeparator or hierarchySpec is missing - required property!";let e=!1;return this.hierarchySpec.reduce(((i,n)=>{const s=this.getCaseValue(n,t),r=e?i:`${i}/${s}`;return"OTHER"===s&&(e=!0),r}),"")}getCase(t){return b(this,void 0,void 0,(function*(){let e;return this.customCases&&(e=this.customCases.find((e=>e.id===t))),e||(e=yield this.context.get(t)),Object.assign(Object.assign({},e),{pathInHierarchy:this.getCaseHierarchyPath(e)})}))}getCaseValue(t,e){switch(t){case"year":return this.getCaseYear(e);case"month":return this.getCaseMonth(e);case"day":return this.getCaseDay(e);case"description":return this.getCaseDescription(e);case"tissues":return this.getCaseTissues(e);case"stains":return this.getCaseStains(e);default:if("id_part_"===t.slice(0,8)&&!isNaN(Number(t.slice(8))))return this.getCaseIdentifierPart(e,Number(t.slice(8)));throw`KeyError[CaseExplorer] "${t}" is not supported!`}}evaluateCaseValue(t,e,i){const n=this.getCaseValue(t,i);switch(t){case"year":return this.evaulateCaseYear(n,e);case"month":return this.evaulateCaseMonth(n,e);case"day":return this.evaulateCaseDay(n,e);case"description":return this.evaluateCaseDescription(n,e);case"tissues":return this.evaluateCaseTissues(n,e);case"stains":return this.evaluateCaseStains(n,e);default:return this.evaulateCaseIdentifierPart(n,e)}}getCaseYear(t){return(e=t.created_at,new Date(1e3*x(e)).getFullYear()).toString();var e}getCaseMonth(t){return(e=t.created_at,new Date(1e3*x(e)).getMonth()).toString();var e}getCaseDay(t){return(e=t.created_at,new Date(1e3*x(e)).getDate()).toString();var e}getCaseIdentifierPart(t,e){if(!this.identifierSeparator)throw"ArgumentError[CaseExplorer] identifierSeparator is missing - required property!";const i=new RegExp(this.identifierSeparator).exec(t.local_id||"");if(!i)return"OTHER";if(e<1||e>=i.length)throw'KeyError[CaseExplorer] invalid key "id_part_", group index is not valid!';return i[e]}getCaseDescription(t){return t.description||""}getCaseTissues(t){return Object.keys(t.tissues)}getCaseStains(t){return Object.keys(t.stains)}evaulateCaseYear(t,e){return t===e}evaulateCaseMonth(t,e){return t===e}evaulateCaseDay(t,e){return t===e}evaulateCaseIdentifierPart(t,e){return t===e}evaluateCaseDescription(t,e){return((t,e)=>{const i=e.split(" ").filter(Boolean).map((t=>`(?=.*\\b${t}\\b)`)),n=new RegExp(i.join(""),"gim");return null!==t.match(n)})(t,e)}evaluateCaseTissues(t,e){return e instanceof Array||(e=[e]),e.every((e=>t.includes(e)))}evaluateCaseStains(t,e){return e instanceof Array||(e=[e]),e.every((e=>t.includes(e)))}hierarchyLevel(t,e,i,n,s){if(e>=t.length)return{levelName:s,lastLevel:!0,items:i.map((t=>Object.assign(Object.assign({},t),{pathInHierarchy:n})))};const r=(o=i=>{const n=this.getCaseValue(t[e],i);return Array.isArray(n)?n[0]||"":n},i.reduce(((t,e)=>{var i;return(t[i=o(e)]||(t[i]=[])).push(e),t}),{}));var o;const a=Object.keys(r).map((i=>{var s;const o=(null===(s=this.hierarchyNameOverrides[t[e]])||void 0===s?void 0:s[i])||i;return"OTHER"===i?this.hierarchyLevel(t,t.length,r[i],`${n}/${o}`,o):this.hierarchyLevel(t,e+1,r[i],`${n}/${o}`,o)}));return{levelName:s,lastLevel:!1,items:a}}hierarchy(){return b(this,void 0,void 0,(function*(){if(!this.caseHierarchy){const t=yield this.getCustomCases();this.caseHierarchy=this.hierarchyLevel(this.hierarchySpec,0,t,"")}return this.caseHierarchy}))}search(t){return b(this,void 0,void 0,(function*(){let e=yield this.getCustomCases();return t.forEach((({key:t,value:i})=>e=e.filter((e=>this.evaluateCaseValue(t,i,e))))),e}))}tissues(t="EN"){return b(this,void 0,void 0,(function*(){if(!this.caseTissues){const e=yield this.getCustomCases(),i=[];e.forEach((e=>Object.entries(e.tissues).map((([e,i])=>({name:e,locName:i[t]}))).forEach((t=>i.push(t))))),this.caseTissues=[...new Map(i.map((t=>[JSON.stringify([t.name,t.locName]),t]))).values()]}return this.caseTissues}))}stains(t="EN"){return b(this,void 0,void 0,(function*(){if(!this.caseStains){const e=yield this.getCustomCases(),i=[];e.forEach((e=>Object.entries(e.stains).map((([e,i])=>({name:e,locName:i[t]}))).forEach((t=>i.push(t))))),this.caseStains=[...new Map(i.map((t=>[JSON.stringify([t.name,t.locName]),t]))).values()]}return this.caseStains}))}}var I=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class S{constructor(t){this.lastCaseId=null,this.data=null,this.slidesData=null,this.masksData=null,this.maskIdentifierSeparator="",this.maskIdentifierValue="",this.context=t}use(t,e){this.maskIdentifierSeparator=t,this.maskIdentifierValue=e}getAllSlides(t){return I(this,void 0,void 0,(function*(){return this.lastCaseId===t&&this.data||(this.data=(yield this.context.slides(t)).items),this.data}))}slides(t){return I(this,void 0,void 0,(function*(){return this.lastCaseId===t&&this.slidesData||(this.slidesData=(yield this.getAllSlides(t)).filter((t=>!w(t.local_id||"",this.maskIdentifierSeparator,1,this.maskIdentifierValue)))),this.slidesData}))}masks(t){return I(this,void 0,void 0,(function*(){return this.lastCaseId===t&&this.masksData||(this.masksData=(yield this.getAllSlides(t)).filter((t=>w(t.local_id||"",this.maskIdentifierSeparator,1,this.maskIdentifierValue)))),this.masksData}))}}var P=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class C extends f{constructor(t){super(),this.data=null,this.context=t,this.caseExplorer=new T(this),this.wsiExplorer=new S(this)}list(){return P(this,void 0,void 0,(function*(){return this.data||(this.data=yield this.context.rawQuery("/cases")),this.data}))}get(t){return P(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/cases/${t}`)}))}slides(t){return P(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/cases/${t}/slides`)}))}}var E=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class $ extends f{constructor(t){super(),this.data=null,this.context=t}create(t,e){return E(this,void 0,void 0,(function*(){const i=this.context;return i.requires("caseId",t),i.rawQuery("/examinations",{method:"PUT",body:{case_id:t,app_id:e}})}))}query(t,e,i){return E(this,void 0,void 0,(function*(){return this.context.rawQuery("/examinations/query",{method:"PUT",body:t,query:{skip:e,limit:i}})}))}get(t){return E(this,void 0,void 0,(function*(){const e=this.context;return e.requires("examinationId",t),e.rawQuery(`/examinations/${t}`)}))}scope(t){return E(this,void 0,void 0,(function*(){const e=this.context;return e.requires("examinationId",t),e.rawQuery(`/examinations/${t}/scope`,{method:"PUT"})}))}}var _=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class k extends v{constructor(t){super(),this.data=null,this.context=t}getRaw(){return _(this,void 0,void 0,(function*(){return this.data||(this.data=yield this.context.rawQuery("/app-ui-storage/user")),this.data}))}flush(){return _(this,void 0,void 0,(function*(){return this.data?yield this.context.rawQuery("/app-ui-storage/user",{method:"PUT",body:this.data}):null}))}get(t){return _(this,void 0,void 0,(function*(){const e=yield this.getRaw();return"string"==typeof e.content[t]?JSON.parse(e.content[t]):e.content[t]}))}set(t,e,i){return _(this,void 0,void 0,(function*(){const n=JSON.stringify(e),s=yield this.getRaw();s.content[t]=n,this.data=s,i&&this.flush()}))}erase(){return _(this,void 0,void 0,(function*(){this.data={content:{}},this.flush()}))}}var O=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class A extends v{constructor(t){super(),this.context=t}upload(t,e={}){return O(this,void 0,void 0,(function*(){return yield this.context.rawQuery("/annotations",{method:"POST",query:e,body:this.data})}))}create(t,e={}){return O(this,void 0,void 0,(function*(){return yield this.context.rawQuery("/annotations",{method:"POST",query:e,body:this.data})}))}delete(t){return O(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/annotations/${t}`,{method:"DELETE"})}))}update(t,e,i={}){return O(this,void 0,void 0,(function*(){return yield this.delete(t),!i.externalIds&&e.id&&(i.externalIds=!0),yield this.create(e,i)}))}}var R=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class U extends v{constructor(t){super(),this.data=null,this.context=t}get(t){return R(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/collections/${t}`)}))}create(t){return R(this,void 0,void 0,(function*(){return yield this.context.rawQuery("/collections",{method:"POST",body:t})}))}delete(t){return R(this,void 0,void 0,(function*(){yield this.context.rawQuery(`/collections/${t}`,{method:"DELETE"})}))}queryItems(t,e){return R(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/collections/${t}/items/query`,{method:"PUT",body:e})}))}createItems(t,e){return R(this,void 0,void 0,(function*(){yield this.context.rawQuery(`/collections/${t}/items`,{method:"POST",body:Object.assign({},e)})}))}deleteItem(t,e){return R(this,void 0,void 0,(function*(){yield this.context.rawQuery(`/collections/${t}/items/${e}`,{method:"DELETE"})}))}}var j=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class N extends v{constructor(t){super(),this.data=null,this.context=t}getJobs(){return j(this,void 0,void 0,(function*(){return(yield this.context.rawQuery("/jobs")).items}))}get(t){return j(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/jobs/${t}`)}))}}var D=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class Q extends v{constructor(t){super(),this.data=null,this.context=t}get(t){return D(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/pixelmaps/${t}`)}))}post(t){return D(this,void 0,void 0,(function*(){return yield this.context.rawQuery("/pixelmaps",{method:"POST",body:t})}))}delete(t){return D(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/pixelmaps/${t}`,{method:"DELETE"})}))}query(t){return D(this,void 0,void 0,(function*(){return(yield this.context.rawQuery("/pixelmaps/query",{method:"PUT",body:t})).items}))}getTile(t,e,i,n){return D(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/pixelmaps/${t}/level/${e}/position/${i}/${n}/data`,{responseType:"blob"})}))}uploadTile(t,e,i,n,s){return D(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/pixelmaps/${t}/level/${e}/position/${i}/${n}/data`,{method:"PUT",body:s})}))}deleteTile(t,e,i,n){return D(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/pixelmaps/${t}/level/${e}/position/${i}/${n}/data`,{method:"DELETE"})}))}bulkGetTile(t,e,i,n,s,r){return D(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/pixelmaps/${t}/level/${e}/position/start/${i}/${n}/end/${s}/${r}/data`,{responseType:"blob"})}))}bulkUploadTile(t,e,i,n,s,r,o){return D(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/pixelmaps/${t}/level/${e}/position/start/${i}/${n}/end/${s}/${r}/data`,{method:"PUT",body:o})}))}}var q=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class M extends y{constructor(t){super(),this.scopeContext=null,this._defaultExaminationId="",this._tokenRefetchInterval=null,this.activeExaminationId="",this.activeCaseId="",this.activeAppId="",this.context=t,this.raw=new a(this.context.options.apiUrl+M.apiPath),this.storage=new k(this),this.annotations=new A(this),this.collections=new U(this),this.jobs=new N(this),this.pixelmaps=new Q(this)}use(t,e=void 0){return q(this,void 0,void 0,(function*(){this.requires("root::userId",this.context.userId);const i=e=>q(this,void 0,void 0,(function*(){let i=yield this.context.examinations.query({apps:[e],creators:[this.context.userId]});if(i.item_count>0){let t=i.items.find((t=>"OPEN"===t.state));if(t)return t}return yield this.context.examinations.create(t,e)}));let n;if(e?n=yield i(e):this._defaultExaminationId&&(n=yield this.context.examinations.get(this._defaultExaminationId)),!n){let t=yield this.context.apps.default();n=yield i(t.app_id),this._defaultExaminationId=n.id}yield this.from(n)}))}get scopeToken(){var t;return null===(t=this.scopeContext)||void 0===t?void 0:t.access_token}from(t){return q(this,void 0,void 0,(function*(){this.reset(),this.scopeContext=yield this.context.examinations.scope(t.id),this.activeCaseId=t.case_id,this.activeAppId=t.app_id,this.activeExaminationId=t.id;const e=u(d(this.scopeContext.access_token));this._tokenRefetchInterval=setInterval((()=>q(this,void 0,void 0,(function*(){this.scopeContext=yield this.context.examinations.scope(t.id)}))),e),this.raiseEvent("init")}))}reset(){this.activeExaminationId="",this.scopeContext=null,this._tokenRefetchInterval&&(clearInterval(this._tokenRefetchInterval),this._tokenRefetchInterval=null,this.raiseEvent("reset"))}rawQuery(t,e){var i,n;return q(this,void 0,void 0,(function*(){this.requires("this.scopeContext",this.scopeContext),(e=e||{}).headers=e.headers||{},e.headers.Authorization=`Bearer ${null===(i=this.scopeContext)||void 0===i?void 0:i.access_token}`,t&&!t.startsWith("/")&&(t=`/${t}`);try{return yield this.raw.http(`/${null===(n=this.scopeContext)||void 0===n?void 0:n.scope_id}${t}`,e)}catch(i){if(401===i.statusCode)return this.scopeContext=yield this.context.examinations.scope(this.activeExaminationId),yield this.raw.http(`/${this.scopeContext.scope_id}${t}`,e);throw i}}))}}M.apiPath="/v3/scopes";const J=M;var H,L,V,F=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class z extends f{constructor(t){super(),this.data=null,this.context=t}slideInfo(t){return F(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/slides/${t}/info`)}))}slideThumbnail(t,e,i,n){return F(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/slides/${t}/thumbnail/max_size/${e}/${i}`,{query:{image_format:n},responseType:"blob"})}))}slideLabel(t,e,i,n){return F(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/slides/${t}/label/max_size/${e}/${i}`,{query:{image_format:n},responseType:"blob"})}))}loadTile(t,e,i,n,s){return F(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/slides/${t}/tile/level/${e}/tile/${i}/${n}`,{query:{image_format:s},responseType:"blob"})}))}}!function(t){t.USER="user",t.SCOPE="scope",t.JOB="job"}(H||(H={})),function(t){t.ANNOTATION="annotation",t.COLLECTION="collection",t.CLASS="class",t.PRIMITIVE="primitive",t.WSI="wsi",t.CASE="case",t.USER="user",t.SCOPE="scope",t.JOB="job"}(L||(L={})),function(t){t.Background="background",t.Params="params",t.Shader="shader",t.Visualization="visualization"}(V||(V={}));var W=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class B{constructor(t){this.data=null,this.defaultSlideMetadata={visualization:{}},this.defaultMaskMetadata={},this.context=t}getWsiMetadataItem(t,e=!0){return W(this,void 0,void 0,(function*(){let i=(yield this.context.query({references:[t]})).find((t=>t.data_type===(e?"slide":"mask")+"_metadata"));return i||(i=yield this.createWsiMetadataItem(t,e?this.defaultSlideMetadata:this.defaultMaskMetadata)),i}))}createWsiMetadataItem(t,e,i=!0){return W(this,void 0,void 0,(function*(){return yield this.context.createValue(e,`Metadata of ${i?"slide":"mask"} ${t}`,void 0,t,L.WSI,(i?"slide":"mask")+"_metadata")}))}getSlideMetadata(t){return W(this,void 0,void 0,(function*(){return JSON.parse((yield this.getWsiMetadataItem(t)).value)}))}updateSlideMetadata(t,e){return W(this,void 0,void 0,(function*(){const i=yield this.getWsiMetadataItem(t);try{const t=yield this.context.update(i.id,Object.assign(Object.assign({},i),{value:JSON.stringify(e)}));return JSON.parse(t.value)}catch(t){return!1}}))}getShadersConfig(t){return W(this,void 0,void 0,(function*(){const e={};for(let i=0;iW(this,void 0,void 0,(function*(){return Object.assign(Object.assign({},yield this.context.visTemplates.getTemplate(V.Visualization,t.visTemplate)),{name:t.name,shaders:yield this.getShadersConfig(t.shaders)})}))))))}}))}getMaskMetadata(t){return W(this,void 0,void 0,(function*(){return JSON.parse((yield this.getWsiMetadataItem(t,!1)).value)}))}}var K=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};const G="vis_templates";class Y{constructor(t){this.data=null,this.context=t}fetchTemplateItem(t,e){return K(this,void 0,void 0,(function*(){return(yield this.context.query({references:[null],data_types:[`${G}_${t}`]})).find((t=>t.name===e))}))}getTemplate(t,e){return K(this,void 0,void 0,(function*(){const i=yield this.fetchTemplateItem(t,e);return!!i&&JSON.parse(i.value)}))}createTemplate(t,e,i){return K(this,void 0,void 0,(function*(){return!(yield this.fetchTemplateItem(t,e))&&(yield this.context.createValue(i,`${e}`,void 0,void 0,void 0,`${G}_${t}`))}))}deleteTemplate(t,e){return K(this,void 0,void 0,(function*(){const i=yield this.fetchTemplateItem(t,e);return!!i&&(yield this.context.delete(i.id),!0)}))}}var X=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class Z{constructor(t){this.data=null,this.presetDataType="annot_presets",this.context=t}use(t){this.presetDataType=t}getPresetsItem(t=!1){return X(this,void 0,void 0,(function*(){if(!this.data||t){let t=(yield this.context.query({references:[null],data_types:[this.presetDataType]})).find((t=>!0));t||(t=yield this.createPresetsItem({presets:[]})),this.data=t}return this.data}))}createPresetsItem(t){return X(this,void 0,void 0,(function*(){return yield this.context.createValue(t,"Global annotation presets",void 0,void 0,void 0,this.presetDataType)}))}getAnnotPresets(t=!1){return X(this,void 0,void 0,(function*(){const e=yield this.getPresetsItem(t);return{presets:JSON.parse(e.value).presets,lastModifiedAt:e.modified_at}}))}mergePresets(t,e,i){const n=[...t];return e.forEach((t=>n.some((e=>e.id===t.id))||!t.createdAt||t.createdAt<=i?null:n.push(t))),n}updateAnnotPresets(t,e,i=!1){return X(this,void 0,void 0,(function*(){const n=yield this.getPresetsItem(!0),s=JSON.parse(n.value).presets;let r=t,o=!0;if(n.modified_at!==e){if(i)return{presets:s,successfulUpdate:!1,lastModifiedAt:n.modified_at};r=this.mergePresets(s,r,e),o=!1}try{const t=yield this.context.update(n.id,Object.assign(Object.assign({},n),{value:JSON.stringify({presets:r})}));return{presets:JSON.parse(t.value).presets,successfulUpdate:o,lastModifiedAt:t.modified_at}}catch(t){if(409===t.statusCode){const t=yield this.updateAnnotPresets(r,n.modified_at);return Object.assign(Object.assign({},t),{successfulUpdate:o})}throw t}}))}deleteAnnotPresets(){return X(this,void 0,void 0,(function*(){const t=yield this.getPresetsItem(!0);yield this.context.delete(t.id),this.data=null}))}}var tt=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class et{constructor(t){this.data=null,this.configDataType="app_job_config",this.context=t}use(t){this.configDataType=t}fetchJobConfigItem(t){return tt(this,void 0,void 0,(function*(){return(yield this.context.query({references:[t],data_types:[this.configDataType]})).find(Boolean)}))}getJobConfig(t){return tt(this,void 0,void 0,(function*(){const e=yield this.fetchJobConfigItem(t);return!!e&&JSON.parse(e.value)}))}createJobConfig(t,e){return tt(this,void 0,void 0,(function*(){return!(yield this.fetchJobConfigItem(t))&&(yield this.context.createValue(e,"Job config of App",void 0,t,L.JOB,this.configDataType))}))}deleteJobConfig(t){return tt(this,void 0,void 0,(function*(){const e=yield this.fetchJobConfigItem(t);return!!e&&(yield this.context.delete(e.id),!0)}))}}var it=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class nt{constructor(t){this.data=null,this.context=t,this.wsiMetadata=new B(this),this.visTemplates=new Y(this),this.annotPresets=new Z(this),this.jobConfig=new et(this)}get(t){return it(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/global-storage/${t}`)}))}getValue(t){return it(this,void 0,void 0,(function*(){const e=yield this.get(t);if("string"===e.type)try{return JSON.parse(e.value)}catch(t){return e.value}return e.value}))}query(t){return it(this,void 0,void 0,(function*(){return(yield this.context.rawQuery("/global-storage/query",{method:"PUT",body:t})).items}))}create(t){return it(this,void 0,void 0,(function*(){return yield this.context.rawQuery("/global-storage",{method:"POST",body:t})}))}createValue(t,e,i,n,s,r){return it(this,void 0,void 0,(function*(){t=JSON.stringify(t);const o={name:e,description:i,creator_id:this.context.userId,creator_type:H.USER,reference_id:n,reference_type:s,type:"string",value:t,data_type:r};return yield this.create(o)}))}update(t,e){return it(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/global-storage/${t}`,{method:"PUT",body:e})}))}updateValue(t,e){return it(this,void 0,void 0,(function*(){const i=yield this.context.rawQuery(`/global-storage/${t}`);return i.value=JSON.stringify(e),yield this.context.rawQuery(`/global-storage/${t}`,{method:"PUT",body:i})}))}delete(t){return it(this,void 0,void 0,(function*(){yield this.context.rawQuery(`/global-storage/${t}`,{method:"DELETE"})}))}}class st{constructor(t){this.context=t,this.globalStorage=new nt(this)}get userId(){return this.context.userId}rawQuery(t,e={}){return i=this,n=void 0,r=function*(){return this.context.rawQuery(`${st.relativeApiPath}${t}`,e)},new((s=void 0)||(s=Promise))((function(t,e){function o(t){try{c(r.next(t))}catch(t){e(t)}}function a(t){try{c(r.throw(t))}catch(t){e(t)}}function c(e){var i;e.done?t(e.value):(i=e.value,i instanceof s?i:new s((function(t){t(i)}))).then(o,a)}c((r=r.apply(i,n||[])).next())}));var i,n,s,r}}st.relativeApiPath="/rationai";const rt=st;var ot=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{c(n.next(t))}catch(t){r(t)}}function a(t){try{c(n.throw(t))}catch(t){r(t)}}function c(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}c((n=n.apply(t,e||[])).next())}))};class at extends p{constructor(t){super(t),this.defaultScopeKey="",this.version="v3",this.rootURI=this.options.apiUrl+at.apiPath,this.raw=new a(this.rootURI),this.rationai=new rt(this),this.apps=new g(this),this.cases=new C(this),this.examinations=new $(this),this.slides=new z(this),this.scopes=new Map}get defaultScope(){return this.scopes.get(this.defaultScopeKey)}newScopeFrom(t){return ot(this,void 0,void 0,(function*(){const e=new J(this);return yield e.from(t),this.scopes.set(t.id,e),""===this.defaultScopeKey&&(this.defaultScopeKey=t.id),e}))}newScopeUse(t,e){return ot(this,void 0,void 0,(function*(){const i=new J(this);return yield i.use(t,e),this.scopes.set(i.activeExaminationId,i),""===this.defaultScopeKey&&(this.defaultScopeKey=i.activeExaminationId),i}))}getScopeFrom(t){return ot(this,void 0,void 0,(function*(){return this.scopes.get(t.id)||(yield this.newScopeFrom(t))}))}getScopeUse(t,e){return ot(this,void 0,void 0,(function*(){const i=[...this.scopes.values()].filter((i=>i.activeCaseId===t&&(!e||i.activeAppId===e)));return i.length>0?i[0]:yield this.newScopeUse(t,e)}))}rawQuery(t,e={}){const i=Object.create(null,{rawQuery:{get:()=>super.rawQuery}});return ot(this,void 0,void 0,(function*(){return yield i.rawQuery.call(this,t,e),(e=e||{}).headers=e.headers||{},e.headers["User-Id"]=this.userId,this.accessToken&&(e.headers.Authorization=e.headers.Authorization||`Bearer ${this.rawToken}`),this.raw.http(t,e)}))}}at.apiPath="/v3";const ct=at;EmpationAPI=e})(); -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"empationapi.js","mappings":"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","sources":["webpack://EmpationAPI/webpack/bootstrap","webpack://EmpationAPI/webpack/runtime/define property getters","webpack://EmpationAPI/webpack/runtime/hasOwnProperty shorthand","webpack://EmpationAPI/webpack/runtime/make namespace object","webpack://EmpationAPI/./src/events.ts","webpack://EmpationAPI/./src/status-codes.ts","webpack://EmpationAPI/./src/base.ts","webpack://EmpationAPI/./src/utils.ts","webpack://EmpationAPI/./src/root.ts","webpack://EmpationAPI/./src/scope.ts","webpack://EmpationAPI/./src/v3/root/apps.ts","webpack://EmpationAPI/./src/v3/extensions/utils.ts","webpack://EmpationAPI/./src/v3/extensions/case-explorer.ts","webpack://EmpationAPI/./src/v3/extensions/wsi-explorer.ts","webpack://EmpationAPI/./src/v3/root/cases.ts","webpack://EmpationAPI/./src/v3/root/examinations.ts","webpack://EmpationAPI/./src/v3/scope/storage.ts","webpack://EmpationAPI/./src/v3/scope/annotations.ts","webpack://EmpationAPI/./src/v3/scope/collections.ts","webpack://EmpationAPI/./src/v3/scope/jobs.ts","webpack://EmpationAPI/./src/v3/scope/pixelmaps.ts","webpack://EmpationAPI/./src/v3/scope/scope.ts","webpack://EmpationAPI/./src/v3/rationai/types/global-data-creator-type.ts","webpack://EmpationAPI/./src/v3/rationai/types/global-item-reference-type.ts","webpack://EmpationAPI/./src/v3/extensions/types/template-type.ts","webpack://EmpationAPI/./src/v3/root/slides.ts","webpack://EmpationAPI/./src/v3/extensions/wsi-metadata.ts","webpack://EmpationAPI/./src/v3/extensions/visualization-templates.ts","webpack://EmpationAPI/./src/v3/extensions/annot-presets.ts","webpack://EmpationAPI/./src/v3/extensions/job-config.ts","webpack://EmpationAPI/./src/v3/rationai/global-storage.ts","webpack://EmpationAPI/./src/v3/rationai/rationai.ts","webpack://EmpationAPI/./src/v3/root/root.ts"],"sourcesContent":["// The require scope\nvar __webpack_require__ = {};\n\n","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","/**\r\n * OpenSeadragon inspired event system for reduction of dependencies and uniform event approach\r\n */\r\n\r\nexport type EventHandler = (event: any) => void;\r\n\r\nexport class EventSource {\r\n  events: { [key: string]: any } = {};\r\n\r\n  /**\r\n   * Add an event handler to be triggered only once (or a given number of times)\r\n   * for a given event. It is not removable with removeHandler().\r\n   * @function\r\n   * @param {String} eventName - Name of event to register.\r\n   * @param {EventHandler} handler - Function to call when event\r\n   * is triggered.\r\n   * @param {Object} [userData=null] - Arbitrary object to be passed unchanged\r\n   * to the handler.\r\n   * @param {Number} [times=1] - The number of times to handle the event\r\n   * before removing it.\r\n   * @param {Number} [priority=0] - Handler priority. By default, all priorities are 0. Higher number = priority.\r\n   */\r\n  addOnceHandler(\r\n    eventName: string,\r\n    handler: EventHandler,\r\n    userData: object,\r\n    times: number,\r\n    priority: number,\r\n  ) {\r\n    const self = this;\r\n    times = times || 1;\r\n    let count = 0;\r\n    const onceHandler = function (event: object) {\r\n      count++;\r\n      if (count === times) {\r\n        self.removeHandler(eventName, onceHandler);\r\n      }\r\n      return handler(event);\r\n    };\r\n    this.addHandler(eventName, onceHandler, userData, priority);\r\n  }\r\n\r\n  /**\r\n   * Add an event handler for a given event.\r\n   * @function\r\n   * @param {String} eventName - Name of event to register.\r\n   * @param {EventHandler} handler - Function to call when event is triggered.\r\n   * @param {Object} [userData=null] - Arbitrary object to be passed unchanged to the handler.\r\n   * @param {Number} [priority=0] - Handler priority. By default, all priorities are 0. Higher number = priority.\r\n   */\r\n  addHandler(\r\n    eventName: string,\r\n    handler: EventHandler,\r\n    userData: object | null = null,\r\n    priority: number = 0,\r\n  ) {\r\n    let events = this.events[eventName];\r\n    if (!events) {\r\n      this.events[eventName] = events = [];\r\n    }\r\n    if (handler && EventSource.isFunction(handler)) {\r\n      let index = events.length,\r\n        event = {\r\n          handler: handler,\r\n          userData: userData || null,\r\n          priority: priority || 0,\r\n        };\r\n      events[index] = event;\r\n      while (index > 0 && events[index - 1].priority < events[index].priority) {\r\n        events[index] = events[index - 1];\r\n        events[index - 1] = event;\r\n        index--;\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Remove a specific event handler for a given event.\r\n   * @function\r\n   * @param {String} eventName - Name of event for which the handler is to be removed.\r\n   * @param {EventHandler} handler - Function to be removed.\r\n   */\r\n  removeHandler(eventName: string, handler: EventHandler) {\r\n    const events = this.events[eventName],\r\n      handlers: EventHandler[] = [];\r\n    if (!events) {\r\n      return;\r\n    }\r\n    if (Array.isArray(events)) {\r\n      for (let i = 0; i < events.length; i++) {\r\n        if (events[i].handler !== handler) {\r\n          handlers.push(events[i]);\r\n        }\r\n      }\r\n      this.events[eventName] = handlers;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Get the amount of handlers registered for a given event.\r\n   * @param {String} eventName - Name of event to inspect.\r\n   * @returns {number} amount of events\r\n   */\r\n  numberOfHandlers(eventName: string) {\r\n    const events = this.events[eventName];\r\n    if (!events) {\r\n      return 0;\r\n    }\r\n    return events.length;\r\n  }\r\n\r\n  /**\r\n   * Remove all event handlers for a given event type. If no type is given all\r\n   * event handlers for every event type are removed.\r\n   * @function\r\n   * @param {String} eventName - Name of event for which all handlers are to be removed.\r\n   */\r\n  removeAllHandlers(eventName: string) {\r\n    if (eventName) {\r\n      this.events[eventName] = [];\r\n    } else {\r\n      for (const eventType in this.events) {\r\n        this.events[eventType] = [];\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Get a function which iterates the list of all handlers registered for a given event, calling the handler for each.\r\n   * @function\r\n   * @param {String} eventName - Name of event to get handlers for.\r\n   */\r\n  getHandler(eventName: string) {\r\n    let events = this.events[eventName];\r\n    if (!events || !events.length) {\r\n      return null;\r\n    }\r\n    events = events.length === 1 ? [events[0]] : Array.apply(null, events);\r\n    return function (source: any, args: any) {\r\n      let i,\r\n        length = events.length;\r\n      for (i = 0; i < length; i++) {\r\n        if (events[i]) {\r\n          args.eventSource = source;\r\n          args.userData = events[i].userData;\r\n          events[i].handler(args);\r\n        }\r\n      }\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Get a function which iterates the list of all handlers registered for a given event,\r\n   * calling the handler for each and awaiting async ones.\r\n   * @function\r\n   * @param {String} eventName - Name of event to get handlers for.\r\n   */\r\n  getAwaitingHandler(eventName) {\r\n    let events = this.events[eventName];\r\n    if (!events || !events.length) {\r\n      return null;\r\n    }\r\n    events = events.length === 1 ? [events[0]] : Array.apply(null, events);\r\n\r\n    return function (source, args) {\r\n      // We return a promise that gets resolved after all the events finish.\r\n      // Returning loop result is not correct, loop promises chain dynamically\r\n      // and outer code could process finishing logics in the middle of event loop.\r\n      return new Promise((resolve) => {\r\n        const length = events.length;\r\n        function loop(index) {\r\n          if (index >= length || !events[index]) {\r\n            resolve('Resolved!');\r\n            return null;\r\n          }\r\n          args.eventSource = source;\r\n          args.userData = events[index].userData;\r\n          let result = events[index].handler(args);\r\n          result =\r\n            !result || EventSource.type(result) !== 'promise'\r\n              ? Promise.resolve()\r\n              : result;\r\n          return result.then(() => loop(index + 1));\r\n        }\r\n        loop(0);\r\n      });\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Trigger an event, optionally passing additional information.\r\n   * @function\r\n   * @param {String} eventName - Name of event to register.\r\n   * @param {Object} eventArgs - Event-specific data.\r\n   */\r\n  raiseEvent(eventName: string, eventArgs?: any) {\r\n    const handler = this.getHandler(eventName);\r\n    if (handler) {\r\n      return handler(this, eventArgs || {});\r\n    }\r\n    return undefined;\r\n  }\r\n\r\n  /**\r\n   * Trigger an event, optionally passing additional information.\r\n   * This events awaits every asynchronous or promise-returning function.\r\n   * @param {String} eventName - Name of event to register.\r\n   * @param {Object} eventArgs - Event-specific data.\r\n   * @return {Promise|undefined} - Promise resolved upon the event completion.\r\n   */\r\n  raiseEventAwaiting(eventName, eventArgs) {\r\n    //uncomment if you want to get a log of all events\r\n    //$.console.log( \"Awaiting event fired:\", eventName );\r\n\r\n    const awaitingHandler = this.getAwaitingHandler(eventName);\r\n    if (awaitingHandler) {\r\n      return awaitingHandler(this, eventArgs || {});\r\n    }\r\n    return Promise.resolve('No handler for this event registered.');\r\n  }\r\n\r\n  static class2type = {\r\n    '[object Boolean]': 'boolean',\r\n    '[object Number]': 'number',\r\n    '[object String]': 'string',\r\n    '[object Function]': 'function',\r\n    '[object AsyncFunction]': 'function',\r\n    '[object Promise]': 'promise',\r\n    '[object Array]': 'array',\r\n    '[object Date]': 'date',\r\n    '[object RegExp]': 'regexp',\r\n    '[object Object]': 'object',\r\n  };\r\n\r\n  /**\r\n   * Taken from jQuery 1.6.1\r\n   * @function isFunction\r\n   * @memberof OpenSeadragon\r\n   * @see {@link http://www.jquery.com/ jQuery}\r\n   */\r\n  static isFunction(obj: object) {\r\n    return this.type(obj) === 'function';\r\n  }\r\n\r\n  /**\r\n   * Taken from jQuery 1.6.1\r\n   * @function type\r\n   * @memberof OpenSeadragon\r\n   * @see {@link http://www.jquery.com/ jQuery}\r\n   */\r\n  static type(obj: object) {\r\n    return obj === null || obj === undefined\r\n      ? String(obj)\r\n      : this.class2type[obj.toString() as keyof typeof this.class2type] ||\r\n          (typeof obj === 'function' ? 'function' : 'object');\r\n  }\r\n}\r\n","// Source: https://github.com/prettymuchbryce/http-status-codes/blob/master/src/status-codes.ts\r\nexport const STATUS_CODES = {\r\n  100: 'Continue',\r\n  101: 'Switching protocols',\r\n  102: 'Processing',\r\n  103: 'Early Hints',\r\n  200: 'Ok',\r\n  201: 'Created',\r\n  202: 'Accepted',\r\n  203: 'Non Authoritative Information',\r\n  204: 'No Content',\r\n  205: 'Reset Content',\r\n  206: 'Partial Content',\r\n  207: 'Multi Status',\r\n  300: 'Multiple Choices',\r\n  301: 'Moved Permanently',\r\n  302: 'Moved Temporarily',\r\n  303: 'See Other',\r\n  304: 'Not Modified',\r\n  305: 'Use Proxy',\r\n  307: 'Temporary Redirect',\r\n  308: 'Permanent Redirect',\r\n  400: 'Bad Request',\r\n  401: 'Unauthorized',\r\n  402: 'Payment Required',\r\n  403: 'Forbidden',\r\n  404: 'Not Found',\r\n  405: 'Method Not Allowed',\r\n  406: 'Not Acceptable',\r\n  407: 'Proxy Authentication Required',\r\n  408: 'Request Timeout',\r\n  409: 'Conflict',\r\n  410: 'Gone',\r\n  411: 'Length Required',\r\n  412: 'Precondition Failed',\r\n  413: 'Request Too Long',\r\n  414: 'Request Uri Too Long',\r\n  415: 'Unsupported Media Type',\r\n  416: 'Requested Range Not Satisfiable',\r\n  417: 'Expectation Failed',\r\n  /**\r\n   * Official Documentation @ https://tools.ietf.org/html/rfc2324#section-2.3.2\r\n   *\r\n   * Any attempt to brew coffee with a teapot should result in the error code \"418 I'm a teapot\". The resulting entity body May be short and stout.\r\n   */\r\n  418: 'Im A Teapot',\r\n  419: 'Insufficient Space On Resource',\r\n  420: 'Method Failure',\r\n  421: 'Misdirected Request',\r\n  422: 'Unprocessable Entity',\r\n  423: 'Locked',\r\n  424: 'Failed Dependency',\r\n  426: 'Upgrade Required',\r\n  428: 'Precondition Required',\r\n  429: 'Too Many Requests',\r\n  431: 'Request Header Fields Too Large',\r\n  451: 'Unavailable For Legal Reasons',\r\n  500: 'Internal Server Error',\r\n  501: 'Not Implemented',\r\n  502: 'Bad Gateway',\r\n  503: 'Service Unavailable',\r\n  504: 'Gateway Timeout',\r\n  505: 'Http Version Not Supported',\r\n  507: 'Insufficient Storage',\r\n  511: 'Network Authentication Required',\r\n};\r\n","import { EventSource } from './events';\r\nimport { STATUS_CODES } from './status-codes';\r\n\r\nexport interface EmpationAPIOptions {\r\n  anonymousUserId?: string;\r\n  workbenchApiUrl: string;\r\n  apiRootPath?: string;\r\n}\r\n\r\ntype ResponseType = 'json' | 'blob' | 'text';\r\n\r\nexport interface RawOptions {\r\n  body?: object | string;\r\n  query?: any;\r\n  method?: string;\r\n  headers?: { [key: string]: string };\r\n  responseType?: ResponseType;\r\n}\r\n\r\n//https://gist.github.com/TooTallNate/4fd641f820e1325695487dfd883e5285\r\nfunction httpErrorToName(code: number): string {\r\n  const suffix =\r\n    ((code / 100) | 0) === 4 || ((code / 100) | 0) === 5 ? 'error' : '';\r\n  let name = ` ${String(STATUS_CODES[code as keyof typeof STATUS_CODES] || `HTTP Code ${code}`).replace(/error$/i, '')} ${suffix}`;\r\n  return name\r\n    .split(' ')\r\n    .reduce(\r\n      (acc, c) => acc + (c ? c.charAt(0).toUpperCase() + c.slice(1) : ''),\r\n    );\r\n}\r\n\r\nexport class HTTPError extends Error {\r\n  statusCode: number;\r\n  [key: string]: any;\r\n\r\n  public constructor(\r\n    code: number,\r\n    message: string,\r\n    extras?: Record<string, any>,\r\n  ) {\r\n    super(\r\n      message ||\r\n        STATUS_CODES[code as keyof typeof STATUS_CODES] ||\r\n        `HTTP Code ${code}`,\r\n    );\r\n    if (arguments.length >= 3 && extras) {\r\n      // noinspection TypeScriptValidateTypes\r\n      Object.assign(this, extras);\r\n    }\r\n    this.name = httpErrorToName(code);\r\n    this.statusCode = code;\r\n  }\r\n}\r\n\r\nexport interface RawApiOptions {}\r\n\r\nexport class RawAPI {\r\n  public url: string;\r\n  public options: RawOptions;\r\n  constructor(url: string, options: RawApiOptions = {}) {\r\n    this.url = url;\r\n    this.options = options;\r\n  }\r\n\r\n  private _parseQueryParams(params: string | { [key: string]: string }) {\r\n    if (params) {\r\n      if (typeof params === 'string') return params;\r\n\r\n      //cleanup plain objects\r\n      if (params.constructor === Object || params.constructor === undefined) {\r\n        for (let k in params) {\r\n          const v = params[k];\r\n          if (v === null || v === undefined) delete params[k];\r\n        }\r\n      }\r\n      return `?${new URLSearchParams(params)}`;\r\n    }\r\n    return '';\r\n  }\r\n\r\n  private async _fetch(url: string, options: RawOptions): Promise<any> {\r\n    const response = await fetch(url, {\r\n      method: options.method,\r\n      headers: options.headers,\r\n      body: options.body,\r\n    } as RequestInit);\r\n\r\n    let result;\r\n    try {\r\n      result = await response[options.responseType || 'json']();\r\n    } catch (e) {\r\n      throw new HTTPError(\r\n        500,\r\n        `Failed to parse response data. Original status: ${response.status} | ${response.statusText}`,\r\n        {\r\n          url: url,\r\n          error: e,\r\n        },\r\n      );\r\n    }\r\n\r\n    if (!response.ok) {\r\n      throw new HTTPError(response.status, response.statusText, result);\r\n    }\r\n    return result;\r\n  }\r\n\r\n  async http(endpoint: string, options: RawOptions): Promise<any> {\r\n    const hasBody = !!options.body;\r\n    options.method = options.method || (hasBody ? 'POST' : 'GET');\r\n    if (!endpoint.startsWith('/')) {\r\n      endpoint = `/${endpoint}`;\r\n    }\r\n    options.query = this._parseQueryParams(options.query);\r\n    options.headers = options.headers || {};\r\n    options.headers['Content-Type'] = 'application/json';\r\n    if (options.body && typeof options.body !== 'string') {\r\n      options.body = JSON.stringify(options.body);\r\n    } else options.body = undefined;\r\n\r\n    return await this._fetch(this.url + endpoint + options.query, options);\r\n  }\r\n}\r\n\r\n//todo consider implementing 'observe' method that registers and watches certain property list..\r\nexport class AbstractAPI extends EventSource {\r\n  //todo try figuring out how to print the class name too\r\n  //https://stackoverflow.com/questions/280389/how-do-you-find-out-the-caller-function-in-javascript\r\n  private getCallerName() {\r\n    // Get stack array\r\n    const orig = Error.prepareStackTrace;\r\n    Error.prepareStackTrace = (error, stack) => stack;\r\n    const { stack } = new Error();\r\n    Error.prepareStackTrace = orig;\r\n\r\n    const caller = stack?.[2];\r\n    return caller ? caller : 'unknown context';\r\n  }\r\n\r\n  requires(name: string, value: any): void {\r\n    if (!value) {\r\n      throw `ArgumentError[${this.getCallerName()}] ${name} is missing - required property!`;\r\n    }\r\n  }\r\n}\r\n","export interface JwtTokenBase {\r\n  sub: string;\r\n  exp: number;\r\n}\r\n\r\nexport interface JwtToken extends JwtTokenBase {\r\n  iat: number;\r\n  jti: string;\r\n  iss: string;\r\n  aud: string;\r\n  typ: string;\r\n  azp: string;\r\n}\r\n\r\nexport function getJwtTokenExpiresTimeout(token: JwtTokenBase) {\r\n  //timeout OR 300 secs\r\n  return token.exp * 1e3 - Date.now() || 300e3;\r\n}\r\n\r\nexport interface ScopeToken extends JwtTokenBase {\r\n  token_id: number;\r\n}\r\n\r\nexport function parseJwtToken(token: string): JwtTokenBase {\r\n  return JSON.parse(atob(token.split('.')[1]));\r\n}\r\n\r\nexport function sleep(ms: number): Promise<void> {\r\n  return new Promise((resolve) => setTimeout(resolve, ms));\r\n}\r\n\r\nexport class Logger {\r\n  static error(...args: any): void {\r\n    console.error('E:EmpationAPI', ...args);\r\n  }\r\n  static warn(...args: any): void {\r\n    console.warn('W:EmpationAPI', ...args);\r\n  }\r\n  static info(...args: any): void {\r\n    console.info('I:EmpationAPI', ...args);\r\n  }\r\n  static debug(...args: any): void {\r\n    console.debug('D:EmpationAPI', ...args);\r\n  }\r\n}\r\n","import { AbstractAPI, EmpationAPIOptions, RawAPI, RawOptions } from './base';\r\nimport { ScopeAPI } from './scope';\r\nimport {\r\n  getJwtTokenExpiresTimeout,\r\n  JwtTokenBase,\r\n  parseJwtToken,\r\n} from './utils';\r\n\r\nexport abstract class RootContext {\r\n  protected abstract context: RootAPI;\r\n  protected abstract data: any;\r\n}\r\n\r\nexport interface RootAPIOptions {\r\n  anonymousUserId: string;\r\n  workbenchApiUrl: string;\r\n  apiUrl: string;\r\n  apiRootPath: string;\r\n}\r\n\r\n// BaseAPI implements AbstractAPI over /v[version]\r\nexport abstract class RootAPI extends AbstractAPI {\r\n  // RawAPI implements access to the http endpoints\r\n  protected abstract raw: RawAPI;\r\n  // default ScopeAPI implements AbstractAPI over /v[version]/scopes\r\n  protected abstract defaultScopeKey: string;\r\n\r\n  // Map of ScopeAPI, that implement AbstractAPI over /v[version]/scopes, allows keeping multiple scopes open at once\r\n  abstract scopes: Map<string, ScopeAPI>;\r\n\r\n  // Properties\r\n  abstract version: string;\r\n  abstract rootURI: string;\r\n  options: RootAPIOptions;\r\n  cached: object;\r\n  accessToken: JwtTokenBase | null = null;\r\n\r\n  private _userId: string;\r\n  private _tokenExpires: number = 0;\r\n  private _rawToken: string = '';\r\n\r\n  protected constructor(options: EmpationAPIOptions) {\r\n    super();\r\n\r\n    if (!options.workbenchApiUrl) {\r\n      throw 'WB Api url is required!';\r\n    }\r\n\r\n    let apiUrl;\r\n    if (!options.apiRootPath) {\r\n      apiUrl = options.workbenchApiUrl;\r\n    } else if (!options.apiRootPath.startsWith('/')) {\r\n      apiUrl = `${options.workbenchApiUrl}/${options.apiRootPath}`;\r\n    } else {\r\n      apiUrl = `${options.workbenchApiUrl}${options.apiRootPath}`;\r\n    }\r\n    if (apiUrl.endsWith('/')) {\r\n      apiUrl = apiUrl.slice(0, -1);\r\n    }\r\n    this.options = {\r\n      apiUrl,\r\n      workbenchApiUrl: options.workbenchApiUrl,\r\n      anonymousUserId: options.anonymousUserId || 'anonymous',\r\n      apiRootPath: options.apiRootPath || '',\r\n    };\r\n    this._userId = this.options.anonymousUserId;\r\n    this.cached = {};\r\n  }\r\n\r\n  /**\r\n   * Change the User actor for the API. Note: the api will\r\n   * reset it's whole state.\r\n   * @param token setup context from object\r\n   * @param withEvent\r\n   */\r\n  from(token: string, withEvent = true): void {\r\n    if (!token) {\r\n      return this.reset();\r\n    }\r\n    this._rawToken = token;\r\n    withEvent = withEvent && !this.accessToken; //fire event only when we configure new session\r\n    this.accessToken = parseJwtToken(token) as JwtTokenBase;\r\n    const tokenTimeout = getJwtTokenExpiresTimeout(this.accessToken);\r\n    this._tokenExpires = Date.now() + tokenTimeout / 2;\r\n    let userId = this.accessToken.sub;\r\n    if (!userId)\r\n      throw 'Invalid User ID! Must be valid string shorter than 50 characters!';\r\n    if (userId.length > 50) {\r\n      console.warn(\r\n        'User ID exceeded 50 characters! Using User ID shortened to first 50 characters!',\r\n      );\r\n      userId = userId.slice(0, 50);\r\n    }\r\n    if (this.userId === userId) return;\r\n    this._userId = userId;\r\n    if (withEvent) this.raiseEvent('init');\r\n  }\r\n\r\n  /**\r\n   * Change the User actor for the API, without providing token. Used in no-token configuration. Note: the api will\r\n   * reset it's whole state.\r\n   * @param token setup context from object\r\n   * @param withEvent\r\n   */\r\n  use(userId: string, withEvent = true): void {\r\n    withEvent = withEvent && !this._userId; //fire event only when we configure new session\r\n    this.reset();\r\n    if (!userId || userId.length > 50)\r\n      throw 'Invalid User ID! Must be valid string shorter than 50 characters!';\r\n    this._userId = userId;\r\n    if (withEvent) this.raiseEvent('init_no_token');\r\n  }\r\n\r\n  reset(): void {\r\n    this._rawToken = '';\r\n    this._tokenExpires = 0;\r\n    this.accessToken = null;\r\n    this._userId = this.options.anonymousUserId;\r\n    this.defaultScopeKey = '';\r\n    this.scopes.forEach((scp) => scp.reset());\r\n    this.scopes.clear();\r\n    this.raiseEvent('reset');\r\n  }\r\n\r\n  get userId(): string {\r\n    return this._userId;\r\n  }\r\n\r\n  get rawToken(): string {\r\n    return this._rawToken;\r\n  }\r\n\r\n  async rawQuery(endpoint: string, options?: RawOptions): Promise<any> {\r\n    if (!this._userId) {\r\n      throw \"User must be configured to access Empaia API: either provide a valid 'anonymous' user ID through env variables, or configure the Root API with a valid token.\";\r\n    }\r\n\r\n    if (this._tokenExpires > 0 && Date.now() > this._tokenExpires) {\r\n      const eventObject = { newToken: '' };\r\n      /**\r\n       * @event token-refresh\r\n       * Awaiting event. Provide newToken value in the event handler argument params object\r\n       * for the root API to consume.\r\n       */\r\n      await this.raiseEventAwaiting('token-refresh', eventObject);\r\n      this.from(eventObject.newToken);\r\n    }\r\n  }\r\n}\r\n","import { AbstractAPI, RawAPI, RawOptions } from './base';\r\n\r\nexport abstract class ScopeContext {\r\n  protected abstract context: ScopeAPI;\r\n  protected abstract data: any;\r\n}\r\n\r\n/**\r\n * Scope Binds Examination and User.\r\n */\r\nexport abstract class ScopeAPI extends AbstractAPI {\r\n  protected abstract raw: RawAPI;\r\n\r\n  /**\r\n   * Change the active Scope for the API. Note: the scope state will reset.\r\n   * @param caseId case id - manages the active scope\r\n   * @param appId manages the active scope through examination<user, app>\r\n   *          if undefined, the examination is managed internally\r\n   */\r\n  abstract use(caseId: string, appId?: string): Promise<void>;\r\n\r\n  /**\r\n   * Change the active Scope for the API. Note: the scope state will reset.\r\n   * @param examination setup context from object\r\n   */\r\n  abstract from(examination: object): Promise<void>;\r\n\r\n  abstract reset(): void;\r\n\r\n  abstract rawQuery(endpoint: string, options?: RawOptions): Promise<any>;\r\n}\r\n","import { RootAPI, RootContext } from '../../root';\r\nimport Root from './root';\r\nimport { AppList } from './types/app-list';\r\nimport { AppQuery } from './types/app-query';\r\nimport { AppOutput } from './types/app-output';\r\n\r\nexport default class Apps extends RootContext {\r\n  protected context: RootAPI;\r\n  protected data: AppList | null = null;\r\n  private _defaultApp: AppOutput | null = null;\r\n\r\n  constructor(context: Root) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  async list(): Promise<AppList> {\r\n    return (this.data = (await this.context.rawQuery('/apps/query', {\r\n      method: 'PUT',\r\n      body: {\r\n        apps: null,\r\n        tissues: null,\r\n        stains: null,\r\n        job_modes: null,\r\n      },\r\n    })) as AppList);\r\n  }\r\n\r\n  async query(query: AppQuery): Promise<AppList> {\r\n    return (this.data = (await this.context.rawQuery('/apps/query', {\r\n      method: 'PUT',\r\n      body: query,\r\n    })) as AppList);\r\n  }\r\n\r\n  async default(): Promise<AppOutput> {\r\n    if (!this.data) await this.list();\r\n    for (let app of this.data!.items) {\r\n      if (app.name_short === 'MAP3' && app.vendor_name === 'rationai') {\r\n        this._defaultApp = app;\r\n        break;\r\n      }\r\n    }\r\n    if (!this._defaultApp)\r\n      throw 'Default APP not present in the infrastructure! Was it imported?';\r\n    return this._defaultApp;\r\n  }\r\n}\r\n","/**\r\n * Use for converting strings representing number to a number.\r\n * @param variable Number or string representing a number.\r\n */\r\nconst getNumber = (variable: string | number) => {\r\n  if (typeof variable === 'string') {\r\n    variable = Number(variable);\r\n  }\r\n\r\n  return variable;\r\n};\r\n\r\nexport const getYearFromEpochTime = (epochTime: number | string) => {\r\n  return new Date(getNumber(epochTime) * 1000).getFullYear();\r\n};\r\n\r\nexport const getMonthFromEpochTime = (epochTime: number | string) => {\r\n  return new Date(getNumber(epochTime) * 1000).getMonth();\r\n};\r\n\r\nexport const getDayFromEpochTime = (epochTime: number | string) => {\r\n  return new Date(getNumber(epochTime) * 1000).getDate();\r\n};\r\n\r\n/**\r\n * Match string on specific substring and value.\r\n * @param str String value.\r\n * @param separator Regex that should match the provided value.\r\n * @param groupIdx Index of a group matched by the regex.\r\n * @param value Value the matched group should have.\r\n */\r\nexport const matchStringOnSeparatorGroup = (\r\n  str: string,\r\n  separator: string,\r\n  groupIdx: number,\r\n  value: string,\r\n) => {\r\n  const matches = new RegExp(separator).exec(str);\r\n  if (!matches || groupIdx < 1 || groupIdx >= matches.length) return false;\r\n  return matches[groupIdx] === value;\r\n};\r\n\r\n/**\r\n * Match string if it contains some of the specified tokens.\r\n * @param stringToMatch String value.\r\n * @param tokenString String containing tokens/words split by a \" \".\r\n */\r\nexport const matchStringOnTokens = (\r\n  stringToMatch: string,\r\n  tokenString: string,\r\n) => {\r\n  const tokens = tokenString\r\n    .split(' ')\r\n    .filter(Boolean)\r\n    .map((token) => `(?=.*\\\\b${token}\\\\b)`);\r\n\r\n  const searchTermRegex = new RegExp(tokens.join(''), 'gim');\r\n  return stringToMatch.match(searchTermRegex) !== null;\r\n};\r\n\r\n/**\r\n * Group by for javascript array.\r\n * @param arr Array of objects.\r\n * @param key Function to get value from object by which the objects are grouped.\r\n */\r\nexport const groupBy = <T, K extends keyof any>(arr: T[], key: (i: T) => K) =>\r\n  arr.reduce(\r\n    (groups, item) => {\r\n      (groups[key(item)] ||= []).push(item);\r\n      return groups;\r\n    },\r\n    {} as Record<K, T[]>,\r\n  );\r\n","import Cases from '../root/cases';\r\nimport { Case } from '../root/types/case';\r\nimport { CaseH } from './types/case-h';\r\nimport { CaseHierarchy } from './types/case-hierarchy-result';\r\nimport { CaseSearchParams } from './types/case-search-params';\r\nimport {\r\n  getDayFromEpochTime,\r\n  getMonthFromEpochTime,\r\n  getYearFromEpochTime,\r\n  groupBy,\r\n  matchStringOnTokens,\r\n} from './utils';\r\n\r\nexport type CaseTissuesStains = {\r\n  name: string;\r\n  locName: string;\r\n};\r\n\r\nexport type HierarchyNameOverrides = {\r\n  [key: string]: {\r\n    [key: string]: string;\r\n  };\r\n};\r\n\r\nexport default class CaseExplorer {\r\n  protected context: Cases;\r\n  protected customCases: CaseH[] | null = null;\r\n  protected caseHierarchy: CaseHierarchy | null = null;\r\n  protected caseTissues: CaseTissuesStains[] | null = null;\r\n  protected caseStains: CaseTissuesStains[] | null = null;\r\n\r\n  identifierSeparator: string = '';\r\n  hierarchySpec: string[] = [];\r\n  hierarchyNameOverrides: HierarchyNameOverrides = {};\r\n\r\n  constructor(context: Cases) {\r\n    this.context = context;\r\n  }\r\n\r\n  /**\r\n   * Configure CaseExplorer.\r\n   * @param identifierSeparator Regex matching local_id and its parts.\r\n   * @param hierarchySpec Array of hierarchy keys defining hierarchy.\r\n   * @param hierarchyNameOverrides Override specific values of specific keys to use as names in hierarchy.\r\n   */\r\n  use(\r\n    identifierSeparator: string,\r\n    hierarchySpec: string[],\r\n    hierarchyNameOverrides: HierarchyNameOverrides = {},\r\n  ): void {\r\n    this.hierarchySpec = hierarchySpec;\r\n    this.identifierSeparator = identifierSeparator;\r\n    this.hierarchyNameOverrides = hierarchyNameOverrides;\r\n  }\r\n\r\n  /**\r\n   * Returns cases extended with path in the specified hierarchy.\r\n   */\r\n  private async getCustomCases() {\r\n    if (!this.customCases) {\r\n      this.customCases = (await this.context.list()).items.map((caseObj) => {\r\n        return {\r\n          ...caseObj,\r\n          pathInHierarchy: this.getCaseHierarchyPath(caseObj),\r\n        };\r\n      });\r\n    }\r\n    return this.customCases;\r\n  }\r\n\r\n  /**\r\n   * Returns case's path in the specified hierarchy.\r\n   * @param caseObj EMPAIA Case.\r\n   */\r\n  getCaseHierarchyPath(caseObj: Case) {\r\n    if (!this.identifierSeparator || !this.hierarchySpec) {\r\n      throw `ArgumentError[CaseExplorer] identifierSeparator or hierarchySpec is missing - required property!`;\r\n    }\r\n    let pathFinished = false;\r\n    return this.hierarchySpec.reduce((prev, curr) => {\r\n      const val = this.getCaseValue(curr, caseObj);\r\n\r\n      const returnVal = pathFinished ? prev : `${prev}/${val}`;\r\n      if (val === 'OTHER') {\r\n        pathFinished = true;\r\n      }\r\n      return returnVal;\r\n    }, '');\r\n  }\r\n\r\n  /**\r\n   * Returns single case.\r\n   * @param caseId ID of a case.\r\n   */\r\n  async getCase(caseId: string): Promise<CaseH> {\r\n    let caseObj: Case | undefined;\r\n    if (this.customCases) {\r\n      caseObj = this.customCases.find((cs) => cs.id === caseId);\r\n    }\r\n\r\n    if (!caseObj) {\r\n      caseObj = await this.context.get(caseId);\r\n    }\r\n\r\n    return { ...caseObj, pathInHierarchy: this.getCaseHierarchyPath(caseObj) };\r\n  }\r\n\r\n  /**\r\n   * Returns case's value, can be a simple attribute, or some derived value.\r\n   * @param key Key specifying value that can be extracted from a case.\r\n   * @param cs Case object.\r\n   */\r\n  private getCaseValue(key: string, cs: Case) {\r\n    switch (key) {\r\n      case 'year': {\r\n        return this.getCaseYear(cs);\r\n      }\r\n      case 'month': {\r\n        return this.getCaseMonth(cs);\r\n      }\r\n      case 'day': {\r\n        return this.getCaseDay(cs);\r\n      }\r\n      case 'description': {\r\n        return this.getCaseDescription(cs);\r\n      }\r\n      case 'tissues': {\r\n        return this.getCaseTissues(cs);\r\n      }\r\n      case 'stains': {\r\n        return this.getCaseStains(cs);\r\n      }\r\n      default: {\r\n        if (key.slice(0, 8) === 'id_part_' && !isNaN(Number(key.slice(8)))) {\r\n          return this.getCaseIdentifierPart(cs, Number(key.slice(8)));\r\n        }\r\n        throw `KeyError[CaseExplorer] \\\"${key}\\\" is not supported!`;\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Evaluates case's value against a provided value.\r\n   * @param key Key specifying value that can be extracted from a case.\r\n   * @param evalValue Value for comparison/evaluation.\r\n   * @param cs Case object.\r\n   */\r\n  private evaluateCaseValue(\r\n    key: string,\r\n    evalValue: string | string[],\r\n    cs: Case,\r\n  ) {\r\n    const caseValue = this.getCaseValue(key, cs);\r\n    switch (key) {\r\n      case 'year': {\r\n        return this.evaulateCaseYear(caseValue as string, evalValue as string);\r\n      }\r\n      case 'month': {\r\n        return this.evaulateCaseMonth(caseValue as string, evalValue as string);\r\n      }\r\n      case 'day': {\r\n        return this.evaulateCaseDay(caseValue as string, evalValue as string);\r\n      }\r\n      case 'description': {\r\n        return this.evaluateCaseDescription(\r\n          caseValue as string,\r\n          evalValue as string,\r\n        );\r\n      }\r\n      case 'tissues': {\r\n        return this.evaluateCaseTissues(\r\n          caseValue as string[],\r\n          evalValue as string | string[],\r\n        );\r\n      }\r\n      case 'stains': {\r\n        return this.evaluateCaseStains(\r\n          caseValue as string[],\r\n          evalValue as string | string[],\r\n        );\r\n      }\r\n      default: {\r\n        // any invalid key will be caught in getCaseValue call\r\n        return this.evaulateCaseIdentifierPart(\r\n          caseValue as string,\r\n          evalValue as string,\r\n        );\r\n      }\r\n    }\r\n  }\r\n\r\n  private getCaseYear(cs: Case): string {\r\n    return getYearFromEpochTime(cs.created_at).toString();\r\n  }\r\n  private getCaseMonth(cs: Case): string {\r\n    return getMonthFromEpochTime(cs.created_at).toString();\r\n  }\r\n  private getCaseDay(cs: Case): string {\r\n    return getDayFromEpochTime(cs.created_at).toString();\r\n  }\r\n  private getCaseIdentifierPart(cs: Case, partIdx: number): string {\r\n    if (!this.identifierSeparator) {\r\n      throw `ArgumentError[CaseExplorer] identifierSeparator is missing - required property!`;\r\n    }\r\n    const parts = new RegExp(this.identifierSeparator).exec(cs.local_id || '');\r\n    if (!parts) return 'OTHER';\r\n    if (partIdx < 1 || partIdx >= parts.length)\r\n      throw `KeyError[CaseExplorer] invalid key \\\"id_part_<index>\\\", group index is not valid!`;\r\n    return parts[partIdx];\r\n  }\r\n  private getCaseDescription(cs: Case): string {\r\n    return cs.description || '';\r\n  }\r\n  private getCaseTissues(cs: Case): string[] {\r\n    return Object.keys(cs.tissues);\r\n  }\r\n  private getCaseStains(cs: Case): string[] {\r\n    return Object.keys(cs.stains);\r\n  }\r\n\r\n  private evaulateCaseYear(value: string, evalValue: string): boolean {\r\n    return value === evalValue;\r\n  }\r\n  private evaulateCaseMonth(value: string, evalValue: string): boolean {\r\n    return value === evalValue;\r\n  }\r\n  private evaulateCaseDay(value: string, evalValue: string): boolean {\r\n    return value === evalValue;\r\n  }\r\n  private evaulateCaseIdentifierPart(\r\n    value: string,\r\n    evalValue: string,\r\n  ): boolean {\r\n    return value === evalValue;\r\n  }\r\n  private evaluateCaseDescription(value: string, evalValue: string): boolean {\r\n    return matchStringOnTokens(value, evalValue);\r\n  }\r\n  private evaluateCaseTissues(\r\n    value: string[],\r\n    evalValue: string | string[],\r\n  ): boolean {\r\n    if (!(evalValue instanceof Array)) {\r\n      evalValue = [evalValue];\r\n    }\r\n    // ALL searched tissues are present in case\r\n    return evalValue.every((tissue) => value.includes(tissue));\r\n\r\n    // SOME searched tissues are present in case\r\n    // return evalValue.some((tissue) => value.includes(tissue))\r\n  }\r\n  private evaluateCaseStains(\r\n    value: string[],\r\n    evalValue: string | string[],\r\n  ): boolean {\r\n    if (!(evalValue instanceof Array)) {\r\n      evalValue = [evalValue];\r\n    }\r\n    // ALL searched stains are present in case\r\n    return evalValue.every((stain) => value.includes(stain));\r\n\r\n    // SOME searched stains are present in case\r\n    // return evalValue.some((stain) => value.includes(stain))\r\n  }\r\n\r\n  /**\r\n   * Recursively constructs a hierarchy by single levels\r\n   */\r\n  private hierarchyLevel(\r\n    keys: string[],\r\n    keyIdx: number,\r\n    cases: Case[],\r\n    currentHierarchyPath: string,\r\n    name?: string,\r\n  ): CaseHierarchy {\r\n    if (keyIdx >= keys.length) {\r\n      return {\r\n        levelName: name,\r\n        lastLevel: true,\r\n        items: cases.map((caseObj) => {\r\n          return { ...caseObj, pathInHierarchy: currentHierarchyPath };\r\n        }),\r\n      };\r\n    }\r\n    // grouping by array values(tissues, stains) is not expected, but works by grouping on first value of array\r\n    const groups = groupBy(cases, (cs) => {\r\n      const value = this.getCaseValue(keys[keyIdx], cs);\r\n      if (Array.isArray(value)) {\r\n        return value[0] || '';\r\n      }\r\n      return value;\r\n    });\r\n\r\n    const items = Object.keys(groups).map((name) => {\r\n      const overrideName =\r\n        this.hierarchyNameOverrides[keys[keyIdx]]?.[name] || name;\r\n      if (name === 'OTHER') {\r\n        return this.hierarchyLevel(\r\n          keys,\r\n          keys.length,\r\n          groups[name],\r\n          `${currentHierarchyPath}/${overrideName}`,\r\n          overrideName,\r\n        );\r\n      }\r\n      return this.hierarchyLevel(\r\n        keys,\r\n        keyIdx + 1,\r\n        groups[name],\r\n        `${currentHierarchyPath}/${overrideName}`,\r\n        overrideName,\r\n      );\r\n    });\r\n\r\n    return { levelName: name, lastLevel: false, items: items };\r\n  }\r\n\r\n  /**\r\n   * Constructs a hierarchy based on spec configured in the CaseExplorer class.\r\n   */\r\n  async hierarchy(): Promise<CaseHierarchy> {\r\n    if (!this.caseHierarchy) {\r\n      const cases = await this.getCustomCases();\r\n      this.caseHierarchy = this.hierarchyLevel(\r\n        this.hierarchySpec,\r\n        0,\r\n        cases,\r\n        '',\r\n      );\r\n    }\r\n    return this.caseHierarchy;\r\n  }\r\n\r\n  /**\r\n   * Search cases.\r\n   * @param query Search query.\r\n   */\r\n  async search(query: CaseSearchParams[]): Promise<CaseH[]> {\r\n    let filteredCases = await this.getCustomCases();\r\n    query.forEach(\r\n      ({ key, value }) =>\r\n        (filteredCases = filteredCases.filter((cs) =>\r\n          this.evaluateCaseValue(key, value, cs),\r\n        )),\r\n    );\r\n\r\n    return filteredCases;\r\n  }\r\n\r\n  /**\r\n   * Get all tissues in available cases.\r\n   * @param localization Language of tissue names.\r\n   */\r\n  async tissues(localization: string = 'EN'): Promise<CaseTissuesStains[]> {\r\n    if (!this.caseTissues) {\r\n      const cases = await this.getCustomCases();\r\n\r\n      const allTissues: CaseTissuesStains[] = [];\r\n      cases.forEach((c) =>\r\n        Object.entries(c.tissues)\r\n          .map(([tisName, tisValue]: [string, any]) => ({\r\n            name: tisName,\r\n            locName: tisValue[localization],\r\n          }))\r\n          .forEach((t) => allTissues.push(t)),\r\n      );\r\n      this.caseTissues = [\r\n        ...new Map(\r\n          allTissues.map((t) => [JSON.stringify([t.name, t.locName]), t]),\r\n        ).values(),\r\n      ];\r\n    }\r\n    return this.caseTissues;\r\n  }\r\n\r\n  /**\r\n   * Get all stains in available cases.\r\n   * @param localization Language of stains names.\r\n   */\r\n  async stains(localization: string = 'EN'): Promise<CaseTissuesStains[]> {\r\n    if (!this.caseStains) {\r\n      const cases = await this.getCustomCases();\r\n\r\n      const allStains: CaseTissuesStains[] = [];\r\n      cases.forEach((c) =>\r\n        Object.entries(c.stains)\r\n          .map(([stnName, stnValue]: [string, any]) => ({\r\n            name: stnName,\r\n            locName: stnValue[localization],\r\n          }))\r\n          .forEach((s) => allStains.push(s)),\r\n      );\r\n      this.caseStains = [\r\n        ...new Map(\r\n          allStains.map((s) => [JSON.stringify([s.name, s.locName]), s]),\r\n        ).values(),\r\n      ];\r\n    }\r\n    return this.caseStains;\r\n  }\r\n}\r\n","import Cases from '../root/cases';\r\nimport { Slide } from '../root/types/slide';\r\nimport { matchStringOnSeparatorGroup } from './utils';\r\n\r\nexport default class WsiExplorer {\r\n  protected context: Cases;\r\n  protected lastCaseId: string | null = null;\r\n  protected data: Slide[] | null = null;\r\n  protected slidesData: Slide[] | null = null;\r\n  protected masksData: Slide[] | null = null;\r\n\r\n  maskIdentifierSeparator: string = '';\r\n  maskIdentifierValue: string = '';\r\n\r\n  constructor(context: Cases) {\r\n    this.context = context;\r\n  }\r\n\r\n  /**\r\n   * Configure WsiExplorer with regex identifying the local_id part and value this part should contain to distiguish masks and slides.\r\n   * @param maskIdentifierSeparator Regex specifying part of local_id that should identify the WSI type\r\n   * @param maskIdentifierValue Value the local_id part of WSI should have to identify the WSI as mask\r\n   */\r\n  use(maskIdentifierSeparator: string, maskIdentifierValue: string): void {\r\n    this.maskIdentifierSeparator = maskIdentifierSeparator;\r\n    this.maskIdentifierValue = maskIdentifierValue;\r\n  }\r\n\r\n  /**\r\n   * Fetch all WSIs of case\r\n   * @param caseId ID of case\r\n   */\r\n  private async getAllSlides(caseId: string): Promise<Slide[]> {\r\n    if (this.lastCaseId !== caseId || !this.data) {\r\n      this.data = (await this.context.slides(caseId)).items;\r\n    }\r\n    return this.data;\r\n  }\r\n\r\n  /**\r\n   * Fetch all actual slides of case\r\n   * @param caseId ID of case\r\n   */\r\n  async slides(caseId: string): Promise<Slide[]> {\r\n    if (this.lastCaseId !== caseId || !this.slidesData) {\r\n      this.slidesData = (await this.getAllSlides(caseId)).filter((slide) => {\r\n        return !matchStringOnSeparatorGroup(\r\n          slide.local_id || '',\r\n          this.maskIdentifierSeparator,\r\n          1,\r\n          this.maskIdentifierValue,\r\n        );\r\n      });\r\n    }\r\n    return this.slidesData;\r\n  }\r\n\r\n  /**\r\n   * Fetch all masks of a case\r\n   * @param caseId ID of case\r\n   */\r\n  async masks(caseId: string): Promise<Slide[]> {\r\n    if (this.lastCaseId !== caseId || !this.masksData) {\r\n      this.masksData = (await this.getAllSlides(caseId)).filter((slide) => {\r\n        return matchStringOnSeparatorGroup(\r\n          slide.local_id || '',\r\n          this.maskIdentifierSeparator,\r\n          1,\r\n          this.maskIdentifierValue,\r\n        );\r\n      });\r\n    }\r\n    return this.masksData;\r\n  }\r\n}\r\n","import { RootAPI, RootContext } from '../../root';\r\nimport CaseExplorer from '../extensions/case-explorer';\r\nimport WsiExplorer from '../extensions/wsi-explorer';\r\nimport Root from './root';\r\nimport { Case } from './types/case';\r\nimport { CaseList } from './types/case-list';\r\nimport { SlideList } from './types/slide-list';\r\n\r\nexport default class Cases extends RootContext {\r\n  protected context: RootAPI;\r\n  protected data: CaseList | null = null;\r\n\r\n  caseExplorer: CaseExplorer;\r\n  wsiExplorer: WsiExplorer;\r\n\r\n  constructor(context: Root) {\r\n    super();\r\n    this.context = context;\r\n\r\n    this.caseExplorer = new CaseExplorer(this);\r\n    this.wsiExplorer = new WsiExplorer(this);\r\n  }\r\n\r\n  async list(): Promise<CaseList> {\r\n    if (!this.data) {\r\n      this.data = (await this.context.rawQuery('/cases')) as CaseList;\r\n    }\r\n    return this.data;\r\n  }\r\n\r\n  async get(caseId: string): Promise<Case> {\r\n    return await this.context.rawQuery(`/cases/${caseId}`);\r\n  }\r\n\r\n  async slides(caseId: string): Promise<SlideList> {\r\n    return await this.context.rawQuery(`/cases/${caseId}/slides`);\r\n  }\r\n}\r\n","import { RootContext } from '../../root';\r\nimport Root from './root';\r\nimport { WorkbenchServiceApiV3CustomModelsExaminationsExamination } from './types/workbench-service-api-v-3-custom-models-examinations-examination';\r\nimport { ExaminationQuery } from './types/examination-query';\r\nimport { ExaminationList } from './types/examination-list';\r\nimport { ScopeTokenAndScopeId } from './types/scope-token-and-scope-id';\r\n\r\nexport default class Examinations extends RootContext {\r\n  protected context: Root;\r\n  protected data: ExaminationList | null = null;\r\n\r\n  constructor(context: Root) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  async create(\r\n    caseId: string,\r\n    appId: string,\r\n  ): Promise<WorkbenchServiceApiV3CustomModelsExaminationsExamination> {\r\n    const self = this.context;\r\n    self.requires('caseId', caseId);\r\n    return self.rawQuery('/examinations', {\r\n      method: 'PUT',\r\n      body: {\r\n        case_id: caseId,\r\n        app_id: appId,\r\n      },\r\n    });\r\n  }\r\n\r\n  async query(\r\n    query: ExaminationQuery,\r\n    skip?: number | undefined,\r\n    limit?: number | undefined,\r\n  ): Promise<ExaminationList> {\r\n    const self = this.context;\r\n    return self.rawQuery(`/examinations/query`, {\r\n      method: 'PUT',\r\n      body: query,\r\n      query: { skip, limit },\r\n    });\r\n  }\r\n\r\n  async get(\r\n    examinationId: string,\r\n  ): Promise<WorkbenchServiceApiV3CustomModelsExaminationsExamination> {\r\n    const self = this.context;\r\n    self.requires('examinationId', examinationId);\r\n    return self.rawQuery(`/examinations/${examinationId}`);\r\n  }\r\n\r\n  async scope(examinationId: string): Promise<ScopeTokenAndScopeId> {\r\n    const self = this.context;\r\n    self.requires('examinationId', examinationId);\r\n    return self.rawQuery(`/examinations/${examinationId}/scope`, {\r\n      method: 'PUT',\r\n    });\r\n  }\r\n}\r\n","import { ScopeContext } from '../../scope';\r\nimport Scope from './scope';\r\nimport { AppUiStorage } from './types/app-ui-storage';\r\n\r\nexport default class Storage extends ScopeContext {\r\n  protected context: Scope;\r\n  protected data: AppUiStorage | null = null;\r\n\r\n  constructor(context: Scope) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  async getRaw(): Promise<AppUiStorage> {\r\n    if (!this.data) {\r\n      this.data = await this.context.rawQuery('/app-ui-storage/user');\r\n    }\r\n    return this.data as AppUiStorage;\r\n  }\r\n\r\n  async flush(): Promise<AppUiStorage | null> {\r\n    if (!this.data) return null;\r\n\r\n    return (await this.context.rawQuery('/app-ui-storage/user', {\r\n      method: 'PUT',\r\n      body: this.data,\r\n    })) as AppUiStorage;\r\n  }\r\n\r\n  async get(key: string): Promise<any> {\r\n    const data = await this.getRaw();\r\n    if (typeof data.content[key] === 'string') {\r\n      return JSON.parse(data.content[key] as string);\r\n    }\r\n\r\n    return data.content[key];\r\n  }\r\n\r\n  async set(key: string, value: any, flush?: boolean): Promise<void> {\r\n    const valueRaw = JSON.stringify(value);\r\n    const dataRaw = await this.getRaw();\r\n    dataRaw.content[key] = valueRaw;\r\n    this.data = dataRaw;\r\n\r\n    if (flush) {\r\n      this.flush();\r\n    }\r\n  }\r\n\r\n  async erase(): Promise<void> {\r\n    this.data = { content: {} };\r\n    this.flush();\r\n  }\r\n}\r\n","import { ScopeContext } from '../../scope';\r\nimport Scope from './scope';\r\nimport { PointAnnotation } from './types/point-annotation';\r\nimport { LineAnnotation } from './types/line-annotation';\r\nimport { ArrowAnnotation } from './types/arrow-annotation';\r\nimport { CircleAnnotation } from './types/circle-annotation';\r\nimport { RectangleAnnotation } from './types/rectangle-annotation';\r\nimport { PolygonAnnotation } from './types/polygon-annotation';\r\nimport { PostPointAnnotations } from './types/post-point-annotations';\r\nimport { PostLineAnnotations } from './types/post-line-annotations';\r\nimport { PostArrowAnnotations } from './types/post-arrow-annotations';\r\nimport { PostCircleAnnotations } from './types/post-circle-annotations';\r\nimport { PostRectangleAnnotations } from './types/post-rectangle-annotations';\r\nimport { PostPolygonAnnotations } from './types/post-polygon-annotations';\r\nimport { PostPointAnnotation } from './types/post-point-annotation';\r\nimport { PostLineAnnotation } from './types/post-line-annotation';\r\nimport { PostArrowAnnotation } from './types/post-arrow-annotation';\r\nimport { PostCircleAnnotation } from './types/post-circle-annotation';\r\nimport { PostRectangleAnnotation } from './types/post-rectangle-annotation';\r\nimport { PostPolygonAnnotation } from './types/post-polygon-annotation';\r\nimport { AnnotationListResponse } from './types/annotation-list-response';\r\nimport { IdObject } from './types/id-object';\r\n\r\nexport interface PostAnnotationQueryParams {\r\n  isRoi?: boolean;\r\n  externalIds?: boolean;\r\n}\r\n\r\nexport default class Annotations extends ScopeContext {\r\n  protected context: Scope;\r\n  protected data: undefined; //unused\r\n\r\n  constructor(context: Scope) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  /**\r\n   * Post multiple annotations as one. Inreality the same endpoint as create(...)\r\n   */\r\n  async upload(\r\n    data:\r\n      | PostPointAnnotations\r\n      | PostLineAnnotations\r\n      | PostArrowAnnotations\r\n      | PostCircleAnnotations\r\n      | PostRectangleAnnotations\r\n      | PostPolygonAnnotations,\r\n    options: PostAnnotationQueryParams = {},\r\n  ): Promise<AnnotationListResponse> {\r\n    //the same as create but for types its simpler to split\r\n    return await this.context.rawQuery('/annotations', {\r\n      method: 'POST',\r\n      query: options,\r\n      body: this.data,\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Create an annotation.\r\n   */\r\n  async create(\r\n    data:\r\n      | PostPointAnnotation\r\n      | PostLineAnnotation\r\n      | PostArrowAnnotation\r\n      | PostCircleAnnotation\r\n      | PostRectangleAnnotation\r\n      | PostPolygonAnnotation,\r\n    options: PostAnnotationQueryParams = {},\r\n  ): Promise<\r\n    | PointAnnotation\r\n    | LineAnnotation\r\n    | ArrowAnnotation\r\n    | CircleAnnotation\r\n    | RectangleAnnotation\r\n    | PolygonAnnotation\r\n  > {\r\n    return await this.context.rawQuery('/annotations', {\r\n      method: 'POST',\r\n      query: options,\r\n      body: this.data,\r\n    });\r\n  }\r\n\r\n  async delete(id: string): Promise<IdObject> {\r\n    return await this.context.rawQuery(`/annotations/${id}`, {\r\n      method: 'DELETE',\r\n    });\r\n  }\r\n\r\n  async update(\r\n    id: string,\r\n    data:\r\n      | PostPointAnnotation\r\n      | PostLineAnnotation\r\n      | PostArrowAnnotation\r\n      | PostCircleAnnotation\r\n      | PostRectangleAnnotation\r\n      | PostPolygonAnnotation,\r\n    options: PostAnnotationQueryParams = {},\r\n  ): Promise<\r\n    | PointAnnotation\r\n    | LineAnnotation\r\n    | ArrowAnnotation\r\n    | CircleAnnotation\r\n    | RectangleAnnotation\r\n    | PolygonAnnotation\r\n  > {\r\n    await this.delete(id);\r\n\r\n    // update might carry id but user forgot to set external IDs to true\r\n    if (!options.externalIds && data.id) {\r\n      options.externalIds = true;\r\n    }\r\n    return await this.create(data, options);\r\n  }\r\n}\r\n","import { ScopeContext } from '../../scope';\r\nimport Scope from './scope';\r\nimport { Collection } from './types/collection';\r\nimport { ItemQuery } from './types/item-query';\r\nimport { ItemQueryList } from './types/item-query-list';\r\n\r\nexport default class Collections extends ScopeContext {\r\n  protected context: Scope;\r\n  protected data: Collection | null = null;\r\n\r\n  constructor(context: Scope) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  async get(collectionId: string): Promise<Collection> {\r\n    const collection: Collection = await this.context.rawQuery(\r\n      `/collections/${collectionId}`,\r\n    );\r\n\r\n    return collection;\r\n  }\r\n\r\n  async create(collection: Collection): Promise<Collection> {\r\n    const createdCollection: Collection = await this.context.rawQuery(\r\n      `/collections`,\r\n      {\r\n        method: 'POST',\r\n        body: collection,\r\n      },\r\n    );\r\n\r\n    return createdCollection;\r\n  }\r\n\r\n  async delete(collectionId: string): Promise<void> {\r\n    await this.context.rawQuery(`/collections/${collectionId}`, {\r\n      method: 'DELETE',\r\n    });\r\n  }\r\n\r\n  async queryItems(\r\n    collectionId: string,\r\n    query: ItemQuery,\r\n  ): Promise<ItemQueryList> {\r\n    const queryResult: ItemQueryList = await this.context.rawQuery(\r\n      `/collections/${collectionId}/items/query`,\r\n      {\r\n        method: 'PUT',\r\n        body: query,\r\n      },\r\n    );\r\n\r\n    return queryResult;\r\n  }\r\n\r\n  async createItems(collectionId: string, items: any): Promise<void> {\r\n    await this.context.rawQuery(`/collections/${collectionId}/items`, {\r\n      method: 'POST',\r\n      body: {\r\n        ...items,\r\n      },\r\n    });\r\n  }\r\n\r\n  async deleteItem(collectionId: string, itemId: string): Promise<void> {\r\n    await this.context.rawQuery(\r\n      `/collections/${collectionId}/items/${itemId}`,\r\n      {\r\n        method: 'DELETE',\r\n      },\r\n    );\r\n  }\r\n}\r\n","import { ScopeContext } from '../../scope';\r\nimport Scope from './scope';\r\nimport { Job } from './types/job';\r\nimport { JobList } from './types/job-list';\r\n\r\nexport default class Jobs extends ScopeContext {\r\n  protected context: Scope;\r\n  protected data: Job[] | null = null;\r\n\r\n  constructor(context: Scope) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  async getJobs(): Promise<Job[]> {\r\n    return ((await this.context.rawQuery('/jobs')) as JobList).items;\r\n  }\r\n\r\n  async get(jobId: string): Promise<Job> {\r\n    return (await this.context.rawQuery(`/jobs/${jobId}`)) as Job;\r\n  }\r\n}\r\n","import { ScopeContext } from '../../scope';\r\nimport Scope from './scope';\r\nimport { ContinuousPixelmap } from './types/continuous-pixelmap';\r\nimport { DiscretePixelmap } from './types/discrete-pixelmap';\r\nimport { IdObject } from './types/id-object';\r\nimport { NominalPixelmap } from './types/nominal-pixelmap';\r\nimport { PixelmapList } from './types/pixelmap-list';\r\nimport { PixelmapQuery } from './types/pixelmap-query';\r\n\r\ntype Pixelmap = ContinuousPixelmap | DiscretePixelmap | NominalPixelmap;\r\n\r\nexport default class Pixelmaps extends ScopeContext {\r\n  protected context: Scope;\r\n  protected data: Pixelmap | null = null;\r\n\r\n  constructor(context: Scope) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  async get(pixelmapId: string): Promise<Pixelmap> {\r\n    return (await this.context.rawQuery(\r\n      `/pixelmaps/${pixelmapId}`,\r\n    )) as Pixelmap;\r\n  }\r\n\r\n  async post(\r\n    pixelmaps: Pixelmaps,\r\n  ): Promise<\r\n    ContinuousPixelmap | DiscretePixelmap | NominalPixelmap | PixelmapList\r\n  > {\r\n    return await this.context.rawQuery(`/pixelmaps`, {\r\n      method: 'POST',\r\n      body: pixelmaps,\r\n    });\r\n  }\r\n\r\n  async delete(pixelmapId: string): Promise<IdObject> {\r\n    return (await this.context.rawQuery(`/pixelmaps/${pixelmapId}`, {\r\n      method: 'DELETE',\r\n    })) as IdObject;\r\n  }\r\n\r\n  async query(query: PixelmapQuery): Promise<Pixelmap[]> {\r\n    const queryResult = (await this.context.rawQuery(`/pixelmaps/query`, {\r\n      method: 'PUT',\r\n      body: query,\r\n    })) as PixelmapList;\r\n    return queryResult.items;\r\n  }\r\n\r\n  async getTile(\r\n    pixelmapId: string,\r\n    level: number,\r\n    tileX: number,\r\n    tileY: number,\r\n  ): Promise<Blob> {\r\n    const tile = await this.context.rawQuery(\r\n      `/pixelmaps/${pixelmapId}/level/${level}/position/${tileX}/${tileY}/data`,\r\n      {\r\n        responseType: 'blob',\r\n      },\r\n    );\r\n    return tile;\r\n  }\r\n\r\n  async uploadTile(\r\n    pixelmapId: string,\r\n    level: number,\r\n    tileX: number,\r\n    tileY: number,\r\n    tile: Blob,\r\n  ): Promise<string> {\r\n    const result = await this.context.rawQuery(\r\n      `/pixelmaps/${pixelmapId}/level/${level}/position/${tileX}/${tileY}/data`,\r\n      {\r\n        method: 'PUT',\r\n        body: tile,\r\n      },\r\n    );\r\n    return result;\r\n  }\r\n\r\n  async deleteTile(\r\n    pixelmapId: string,\r\n    level: number,\r\n    tileX: number,\r\n    tileY: number,\r\n  ): Promise<string> {\r\n    const result = await this.context.rawQuery(\r\n      `/pixelmaps/${pixelmapId}/level/${level}/position/${tileX}/${tileY}/data`,\r\n      {\r\n        method: 'DELETE',\r\n      },\r\n    );\r\n    return result;\r\n  }\r\n\r\n  async bulkGetTile(\r\n    pixelmapId: string,\r\n    level: number,\r\n    startX: number,\r\n    startY: number,\r\n    endX: number,\r\n    endY: number,\r\n  ): Promise<Blob> {\r\n    const tiles = await this.context.rawQuery(\r\n      `/pixelmaps/${pixelmapId}/level/${level}/position/start/${startX}/${startY}/end/${endX}/${endY}/data`,\r\n      {\r\n        responseType: 'blob',\r\n      },\r\n    );\r\n    return tiles;\r\n  }\r\n\r\n  async bulkUploadTile(\r\n    pixelmapId: string,\r\n    level: number,\r\n    startX: number,\r\n    startY: number,\r\n    endX: number,\r\n    endY: number,\r\n    tiles: Blob,\r\n  ): Promise<string> {\r\n    const result = await this.context.rawQuery(\r\n      `/pixelmaps/${pixelmapId}/level/${level}/position/start/${startX}/${startY}/end/${endX}/${endY}/data`,\r\n      {\r\n        method: 'PUT',\r\n        body: tiles,\r\n      },\r\n    );\r\n    return result;\r\n  }\r\n}\r\n","import { ScopeAPI } from '../../scope';\r\nimport { RawAPI, RawOptions } from '../../base';\r\nimport { ScopeTokenAndScopeId } from '../root/types/scope-token-and-scope-id';\r\nimport { WorkbenchServiceApiV3CustomModelsExaminationsExamination } from '../root/types/workbench-service-api-v-3-custom-models-examinations-examination';\r\nimport Root from '../root/root';\r\nimport Storage from './storage';\r\nimport {\r\n  getJwtTokenExpiresTimeout,\r\n  parseJwtToken,\r\n  ScopeToken,\r\n} from '../../utils';\r\nimport Annotations from './annotations';\r\nimport Collections from './collections';\r\nimport Jobs from './jobs';\r\nimport Pixelmaps from \"./pixelmaps\";\r\n\r\nexport default class Scope extends ScopeAPI {\r\n  static apiPath = '/v3/scopes';\r\n\r\n  // Interface\r\n  raw: RawAPI;\r\n  context: Root;\r\n  storage: Storage;\r\n  annotations: Annotations;\r\n  collections: Collections;\r\n  jobs: Jobs;\r\n  pixelmaps: Pixelmaps;\r\n\r\n  // Additional\r\n  scopeContext: ScopeTokenAndScopeId | null = null;\r\n  private _defaultExaminationId: string = '';\r\n  private _tokenRefetchInterval: NodeJS.Timeout | null = null;\r\n\r\n  activeExaminationId: string = '';\r\n  activeCaseId: string = '';\r\n  activeAppId: string = '';\r\n\r\n  constructor(context: Root) {\r\n    super();\r\n    this.context = context;\r\n    this.raw = new RawAPI(this.context.options.apiUrl + Scope.apiPath);\r\n    this.storage = new Storage(this);\r\n    this.annotations = new Annotations(this);\r\n    this.collections = new Collections(this);\r\n    this.jobs = new Jobs(this);\r\n    this.pixelmaps = new Pixelmaps(this);\r\n  }\r\n\r\n  async use(\r\n    caseId: string,\r\n    appId: string | undefined = undefined,\r\n  ): Promise<void> {\r\n    //todo consider caching\r\n\r\n    this.requires('root::userId', this.context.userId);\r\n\r\n    const findExamination = async (\r\n      appId: string,\r\n    ): Promise<WorkbenchServiceApiV3CustomModelsExaminationsExamination> => {\r\n      let examinations = await this.context.examinations.query({\r\n        apps: [appId],\r\n        creators: [this.context.userId],\r\n      });\r\n      if (examinations.item_count > 0) {\r\n        let examination = examinations.items.find((ex) => ex.state === 'OPEN');\r\n        if (examination) return examination;\r\n      }\r\n      return await this.context.examinations.create(caseId, appId);\r\n    };\r\n\r\n    let examination;\r\n    if (appId) {\r\n      examination = await findExamination(appId);\r\n    } else if (this._defaultExaminationId) {\r\n      examination = await this.context.examinations.get(\r\n        this._defaultExaminationId,\r\n      );\r\n    }\r\n\r\n    if (!examination) {\r\n      let app = await this.context.apps.default();\r\n      examination = await findExamination(app.app_id);\r\n      this._defaultExaminationId = examination.id;\r\n    }\r\n\r\n    await this.from(examination);\r\n  }\r\n\r\n  get scopeToken(): string | undefined {\r\n    return this.scopeContext?.access_token;\r\n  }\r\n\r\n  async from(\r\n    examination: WorkbenchServiceApiV3CustomModelsExaminationsExamination,\r\n  ): Promise<void> {\r\n    this.reset();\r\n    this.scopeContext = await this.context.examinations.scope(examination.id);\r\n    this.activeCaseId = examination.case_id;\r\n    this.activeAppId = examination.app_id;\r\n    this.activeExaminationId = examination.id;\r\n    const token = parseJwtToken(this.scopeContext.access_token) as ScopeToken;\r\n    const timeout = getJwtTokenExpiresTimeout(token);\r\n    this._tokenRefetchInterval = setInterval(async () => {\r\n      this.scopeContext = await this.context.examinations.scope(examination.id);\r\n    }, timeout);\r\n    this.raiseEvent('init');\r\n  }\r\n\r\n  reset(): void {\r\n    this.activeExaminationId = '';\r\n    this.scopeContext = null;\r\n    if (this._tokenRefetchInterval) {\r\n      clearInterval(this._tokenRefetchInterval);\r\n      this._tokenRefetchInterval = null;\r\n      //todo clear all cached data\r\n      this.raiseEvent('reset');\r\n    }\r\n  }\r\n\r\n  async rawQuery(endpoint: string, options?: RawOptions): Promise<any> {\r\n    this.requires('this.scopeContext', this.scopeContext);\r\n    options = options || {};\r\n    options.headers = options.headers || {};\r\n    options.headers['Authorization'] =\r\n      `Bearer ${this.scopeContext?.access_token}`;\r\n    if (endpoint && !endpoint.startsWith('/')) {\r\n      endpoint = `/${endpoint}`;\r\n    }\r\n\r\n    try {\r\n      return await this.raw.http(\r\n        `/${this.scopeContext?.scope_id}${endpoint}`,\r\n        options,\r\n      );\r\n    } catch (e) {\r\n      if (e.statusCode === 401) {\r\n        this.scopeContext = await this.context.examinations.scope(\r\n          this.activeExaminationId,\r\n        );\r\n        return await this.raw.http(\r\n          `/${this.scopeContext.scope_id}${endpoint}`,\r\n          options,\r\n        );\r\n      }\r\n\r\n      throw e;\r\n    }\r\n  }\r\n}\r\n","/* tslint:disable */\r\n/* eslint-disable */\r\nexport enum GlobalDataCreatorType {\r\n  USER = 'user',\r\n  SCOPE = 'scope',\r\n  JOB = 'job',\r\n}\r\n","/* tslint:disable */\r\n/* eslint-disable */\r\nexport enum GlobalItemReferenceType {\r\n  ANNOTATION = 'annotation',\r\n  COLLECTION = 'collection',\r\n  CLASS = 'class',\r\n  PRIMITIVE = 'primitive',\r\n  WSI = 'wsi',\r\n  CASE = 'case',\r\n  USER = 'user',\r\n  SCOPE = 'scope',\r\n  JOB = 'job',\r\n}\r\n","/* tslint:disable */\r\n/* eslint-disable */\r\nexport enum TemplateType {\r\n  Background = 'background',\r\n  Params = 'params',\r\n  Shader = 'shader',\r\n  Visualization = 'visualization',\r\n}\r\n","import { RootAPI, RootContext } from '../../root';\r\nimport Root from './root';\r\nimport { Slide } from './types/slide';\r\nimport { SlideInfo } from './types/slide-info';\r\n\r\nexport default class Slides extends RootContext {\r\n  protected context: RootAPI;\r\n  protected data: SlideInfo | null = null;\r\n\r\n  constructor(context: Root) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  async slideInfo(slideId: string): Promise<SlideInfo> {\r\n    return await this.context.rawQuery(`/slides/${slideId}/info`);\r\n  }\r\n\r\n  async slideThumbnail(\r\n    slideId: string,\r\n    maxWidth: number,\r\n    maxHeight: number,\r\n    format?: string,\r\n  ): Promise<Blob> {\r\n    return await this.context.rawQuery(\r\n      `/slides/${slideId}/thumbnail/max_size/${maxWidth}/${maxHeight}`,\r\n      {\r\n        query: {\r\n          image_format: format,\r\n        },\r\n        responseType: 'blob',\r\n      },\r\n    );\r\n  }\r\n\r\n  async slideLabel(\r\n    slideId: string,\r\n    maxWidth: number,\r\n    maxHeight: number,\r\n    format?: string,\r\n  ): Promise<Blob> {\r\n    return await this.context.rawQuery(\r\n      `/slides/${slideId}/label/max_size/${maxWidth}/${maxHeight}`,\r\n      {\r\n        query: {\r\n          image_format: format,\r\n        },\r\n        responseType: 'blob',\r\n      },\r\n    );\r\n  }\r\n\r\n  async loadTile(\r\n    slideId: string,\r\n    level: number,\r\n    x: number,\r\n    y: number,\r\n    format?: string,\r\n  ): Promise<Blob> {\r\n    return await this.context.rawQuery(\r\n      `/slides/${slideId}/tile/level/${level}/tile/${x}/${y}`,\r\n      {\r\n        query: {\r\n          image_format: format,\r\n        },\r\n        responseType: 'blob',\r\n      },\r\n    );\r\n  }\r\n}\r\n","import GlobalStorage from '../rationai/global-storage';\r\nimport { GlobalItem } from '../rationai/types/global-item';\r\nimport { GlobalItemReferenceType } from '../rationai/types/global-item-reference-type';\r\nimport { MaskMetadata } from './types/mask-metadata';\r\nimport { Shader, SlideMetadata } from './types/slide-metadata';\r\nimport { TemplateType } from './types/template-type';\r\n\r\nexport default class WsiMetadata {\r\n  protected context: GlobalStorage;\r\n  protected data: GlobalItem | null = null;\r\n\r\n  private defaultSlideMetadata: SlideMetadata = { visualization: {} };\r\n  private defaultMaskMetadata: MaskMetadata = {};\r\n\r\n  constructor(context: GlobalStorage) {\r\n    this.context = context;\r\n  }\r\n\r\n  // Slide\r\n\r\n  /**\r\n   * Fetch global item containing metadata of WSI.\r\n   * @param wsiId ID of WSI\r\n   * @param isSlide Specify if WSI is a slide or a mask\r\n   */\r\n  private async getWsiMetadataItem(\r\n    wsiId: string,\r\n    isSlide: boolean = true,\r\n  ): Promise<GlobalItem> {\r\n    let metadata = (await this.context.query({ references: [wsiId] })).find(\r\n      (p) => p.data_type === `${isSlide ? 'slide' : 'mask'}_metadata`,\r\n    );\r\n    if (!metadata) {\r\n      metadata = await this.createWsiMetadataItem(\r\n        wsiId,\r\n        isSlide ? this.defaultSlideMetadata : this.defaultMaskMetadata,\r\n      );\r\n    }\r\n    return metadata;\r\n  }\r\n\r\n  /**\r\n   * Create global item containing metadata of WSI.\r\n   * @param wsiId ID of WSI\r\n   * @param value Metadata of WSI\r\n   * @param isSlide Specify if WSI is a slide or a mask\r\n   */\r\n  private async createWsiMetadataItem(\r\n    wsiId: string,\r\n    value: any,\r\n    isSlide: boolean = true,\r\n  ): Promise<GlobalItem> {\r\n    return await this.context.createValue(\r\n      value,\r\n      `Metadata of ${isSlide ? 'slide' : 'mask'} ${wsiId}`,\r\n      undefined,\r\n      wsiId,\r\n      GlobalItemReferenceType.WSI,\r\n      `${isSlide ? 'slide' : 'mask'}_metadata`,\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Get metadata of slide WSI.\r\n   * @param slideId ID of slide WSI\r\n   */\r\n  async getSlideMetadata(slideId: string): Promise<SlideMetadata> {\r\n    return JSON.parse((await this.getWsiMetadataItem(slideId)).value as string);\r\n  }\r\n\r\n  /**\r\n   * Update metadata of slide WSI.\r\n   * @param slideId ID of slide WSI\r\n   * @param value New metadata of WSI\r\n   */\r\n  async updateSlideMetadata(\r\n    slideId: string,\r\n    value: SlideMetadata,\r\n  ): Promise<SlideMetadata | false> {\r\n    const metadataItem = await this.getWsiMetadataItem(slideId);\r\n    try {\r\n      const updatedItem = await this.context.update(metadataItem.id, {\r\n        ...metadataItem,\r\n        value: JSON.stringify(value),\r\n      });\r\n      return JSON.parse(updatedItem.value);\r\n    } catch (e) {\r\n      return false;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Helper method generating shader config for user-created xOpat visualisations\r\n   * @param shaders Array of shaders\r\n   */\r\n  private async getShadersConfig(shaders: Shader[]): Promise<object> {\r\n    const shadersConfig = {};\r\n    for (let i = 0; i < shaders.length; i++) {\r\n      const shaderTemp = await this.context.visTemplates.getTemplate(\r\n        TemplateType.Shader,\r\n        shaders[i].shaderTemplate,\r\n      );\r\n      shadersConfig[shaders[i].id] = {\r\n        ...shaderTemp,\r\n        name: shaders[i].name || shaders[i].id,\r\n        dataReferences: shaders[i].dataRefs,\r\n      };\r\n    }\r\n    return shadersConfig;\r\n  }\r\n\r\n  /**\r\n   * Fetch user-created xOpat visualisations of a slide\r\n   * @param slideId ID of a slide\r\n   */\r\n  async getVisualizations(slideId: string): Promise<any> {\r\n    const slideVis = (await this.getSlideMetadata(slideId)).visualization;\r\n    const visualizations = {\r\n      params: slideVis.paramsTemplate\r\n        ? (await this.context.visTemplates.getTemplate(\r\n            TemplateType.Params,\r\n            slideVis.paramsTemplate,\r\n          )) || undefined\r\n        : undefined,\r\n      data: slideVis.data,\r\n      background: {\r\n        ...(slideVis.background\r\n          ? await this.context.visTemplates.getTemplate(\r\n              TemplateType.Background,\r\n              slideVis.background?.template,\r\n            )\r\n          : {}),\r\n        data: slideVis.background && slideVis.background.dataRef,\r\n      },\r\n      visualizations:\r\n        slideVis.visualizations &&\r\n        (await Promise.all(\r\n          slideVis.visualizations.map(async (vis) => {\r\n            return {\r\n              ...(await this.context.visTemplates.getTemplate(\r\n                TemplateType.Visualization,\r\n                vis.visTemplate,\r\n              )),\r\n              name: vis.name,\r\n              shaders: await this.getShadersConfig(vis.shaders),\r\n            };\r\n          }),\r\n        )),\r\n    };\r\n    return visualizations;\r\n  }\r\n\r\n  // MASK\r\n\r\n  /**\r\n   * Get metadata of mask WSI.\r\n   * @param slideId ID of mask WSI\r\n   */\r\n  async getMaskMetadata(maskId: string): Promise<MaskMetadata> {\r\n    return JSON.parse(\r\n      (await this.getWsiMetadataItem(maskId, false)).value as string,\r\n    );\r\n  }\r\n}\r\n","import GlobalStorage from '../rationai/global-storage';\r\nimport { GlobalItem } from '../rationai/types/global-item';\r\nimport { TemplateType } from './types/template-type';\r\n\r\nconst templatesGlobalItemDataType = 'vis_templates';\r\n\r\nexport default class VisualizationTemplates {\r\n  protected context: GlobalStorage;\r\n  protected data: object | null = null;\r\n\r\n  constructor(context: GlobalStorage) {\r\n    this.context = context;\r\n  }\r\n\r\n  /**\r\n   * Fetch global item containing template.\r\n   * @param type Type of the template (backgroung, params, shader, visualization)\r\n   * @param name Name of the template\r\n   */\r\n  async fetchTemplateItem(\r\n    type: TemplateType,\r\n    name: string,\r\n  ): Promise<GlobalItem | undefined> {\r\n    return (\r\n      await this.context.query({\r\n        references: [null],\r\n        data_types: [`${templatesGlobalItemDataType}_${type}`],\r\n      })\r\n    ).find((item) => item.name === name);\r\n  }\r\n\r\n  /**\r\n   * Fetch template.\r\n   * @param type Type of the template (backgroung, params, shader, visualization)\r\n   * @param name Name of the template\r\n   */\r\n  async getTemplate(type: TemplateType, name: string): Promise<object | false> {\r\n    const tmpl = await this.fetchTemplateItem(type, name);\r\n    if (tmpl) {\r\n      return JSON.parse(tmpl.value);\r\n    }\r\n    return false;\r\n  }\r\n\r\n  /**\r\n   * Create new template.\r\n   * @param type Type of the template (background, params, shader, visualization)\r\n   * @param name Name of the template\r\n   * @param value Value of the template\r\n   */\r\n  async createTemplate(\r\n    type: TemplateType,\r\n    name: string,\r\n    value: object,\r\n  ): Promise<object | false> {\r\n    const existingTmpl = await this.fetchTemplateItem(type, name);\r\n    if (existingTmpl) {\r\n      return false;\r\n    }\r\n    return await this.context.createValue(\r\n      value,\r\n      `${name}`,\r\n      undefined,\r\n      undefined,\r\n      undefined,\r\n      `${templatesGlobalItemDataType}_${type}`,\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Delete template.\r\n   * @param type Type of the template (backgroung, params, shader, visualization)\r\n   * @param name Name of the template\r\n   */\r\n  async deleteTemplate(type: TemplateType, name: string): Promise<boolean> {\r\n    const existingTmpl = await this.fetchTemplateItem(type, name);\r\n    if (!existingTmpl) {\r\n      return false;\r\n    }\r\n    await this.context.delete(existingTmpl.id);\r\n    return true;\r\n  }\r\n}\r\n","import { HTTPError } from '../../../src/base';\r\nimport GlobalStorage from '../rationai/global-storage';\r\nimport { GlobalItem } from '../rationai/types/global-item';\r\nimport { AnnotPreset, AnnotPresetObject } from './types/annot-preset';\r\n\r\ntype AnnotPresetGetResult = {\r\n  presets: AnnotPreset[];\r\n  lastModifiedAt: number;\r\n};\r\n\r\ntype AnnotPresetUpdateResult = {\r\n  presets: AnnotPreset[];\r\n  successfulUpdate: boolean;\r\n  lastModifiedAt: number;\r\n};\r\n\r\nexport default class AnnotPresets {\r\n  protected context: GlobalStorage;\r\n  protected data: GlobalItem | null = null;\r\n\r\n  protected presetDataType: string = 'annot_presets';\r\n\r\n  constructor(context: GlobalStorage) {\r\n    this.context = context;\r\n  }\r\n\r\n  /**\r\n   * Configure AnnotPresets class with data type. Data type is used to filter global items.\r\n   * @param presetDataType Data type of global items used to store annotation presets.\r\n   */\r\n  use(presetDataType: string): void {\r\n    this.presetDataType = presetDataType;\r\n  }\r\n\r\n  /**\r\n   * Fetch global item containing annotation presets (only one should exist).\r\n   * @param fresh Force fresh fetch of global item, otherwise cached version might be used\r\n   */\r\n  private async getPresetsItem(fresh: boolean = false): Promise<GlobalItem> {\r\n    if (!this.data || fresh) {\r\n      let presetsItem = (\r\n        await this.context.query({\r\n          references: [null],\r\n          data_types: [this.presetDataType],\r\n        })\r\n      ).find((item) => true);\r\n      if (!presetsItem) {\r\n        presetsItem = await this.createPresetsItem({ presets: [] });\r\n      }\r\n      this.data = presetsItem;\r\n    }\r\n    return this.data;\r\n  }\r\n\r\n  /**\r\n   * Create global item containing annotation presets (only one should exist).\r\n   * @param value Annotation preset\r\n   */\r\n  private async createPresetsItem(\r\n    value: AnnotPresetObject,\r\n  ): Promise<GlobalItem> {\r\n    return await this.context.createValue(\r\n      value,\r\n      `Global annotation presets`,\r\n      undefined,\r\n      undefined,\r\n      undefined,\r\n      this.presetDataType,\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Get annotation presets.\r\n   * @param fresh Force fresh fetch\r\n   */\r\n  async getAnnotPresets(fresh: boolean = false): Promise<AnnotPresetGetResult> {\r\n    const presetItem = await this.getPresetsItem(fresh);\r\n    return {\r\n      presets: (JSON.parse(presetItem.value as string) as AnnotPresetObject)\r\n        .presets,\r\n      lastModifiedAt: presetItem.modified_at,\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Helper function to merge annotation presets during parallel updates.\r\n   */\r\n  private mergePresets(\r\n    primaryArr: AnnotPreset[],\r\n    secondaryArr: AnnotPreset[],\r\n    localVersion: number,\r\n  ): AnnotPreset[] {\r\n    const newArr = [...primaryArr];\r\n    // item from secondary array is pushed only if item with the same id is not present in primary array, and createdAt date is bigger than localVersion date, meaning item is new\r\n    secondaryArr.forEach((secItem) =>\r\n      newArr.some((primItem) => primItem.id === secItem.id) ||\r\n      !secItem.createdAt ||\r\n      secItem.createdAt <= localVersion\r\n        ? null\r\n        : newArr.push(secItem),\r\n    );\r\n    return newArr;\r\n  }\r\n\r\n  /**\r\n   * Update annotation presets.\r\n   * @param value New presets\r\n   * @param localVersion Local version of presets (modified_at attribute of global item)\r\n   * @param failOnParallelUpdate Force update fail if local version is outdated\r\n   */\r\n  async updateAnnotPresets(\r\n    value: AnnotPreset[],\r\n    localVersion: number,\r\n    failOnParallelUpdate: boolean = false,\r\n  ): Promise<AnnotPresetUpdateResult> {\r\n    // fetch fresh presets\r\n    const remotePresetsItem = await this.getPresetsItem(true);\r\n    const remotePresets = (\r\n      JSON.parse(remotePresetsItem.value as string) as AnnotPresetObject\r\n    ).presets;\r\n\r\n    let localPresets = value;\r\n    let successfulUpdate = true;\r\n\r\n    if (remotePresetsItem.modified_at !== localVersion) {\r\n      if (failOnParallelUpdate) {\r\n        return {\r\n          presets: remotePresets,\r\n          successfulUpdate: false,\r\n          lastModifiedAt: remotePresetsItem.modified_at,\r\n        };\r\n      }\r\n      localPresets = this.mergePresets(\r\n        remotePresets,\r\n        localPresets,\r\n        localVersion,\r\n      );\r\n      successfulUpdate = false;\r\n    }\r\n\r\n    try {\r\n      const updatedItem = await this.context.update(remotePresetsItem.id, {\r\n        ...remotePresetsItem,\r\n        value: JSON.stringify({ presets: localPresets }),\r\n      });\r\n      const updatedPresets = (\r\n        JSON.parse(updatedItem.value) as AnnotPresetObject\r\n      ).presets;\r\n      return {\r\n        presets: updatedPresets,\r\n        successfulUpdate: successfulUpdate,\r\n        lastModifiedAt: updatedItem.modified_at,\r\n      };\r\n    } catch (e) {\r\n      if ((e as HTTPError).statusCode === 409) {\r\n        const retryAttempt = await this.updateAnnotPresets(\r\n          localPresets,\r\n          remotePresetsItem.modified_at,\r\n        );\r\n        return { ...retryAttempt, successfulUpdate: successfulUpdate };\r\n      }\r\n      throw e;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Delete annotation presets.\r\n   */\r\n  async deleteAnnotPresets(): Promise<void> {\r\n    const presetsItem = await this.getPresetsItem(true);\r\n    await this.context.delete(presetsItem.id);\r\n    this.data = null;\r\n  }\r\n}\r\n","import GlobalStorage from '../rationai/global-storage';\r\nimport { GlobalItem } from '../rationai/types/global-item';\r\nimport { GlobalItemReferenceType } from '../rationai/types/global-item-reference-type';\r\n\r\nexport default class JobConfig {\r\n  protected context: GlobalStorage;\r\n  protected data: object | null = null;\r\n\r\n  protected configDataType: string = 'app_job_config';\r\n\r\n  constructor(context: GlobalStorage) {\r\n    this.context = context;\r\n  }\r\n\r\n  /**\r\n   * Configure JobConfig class with data type. Data type is used to filter global items.\r\n   * @param configDataType Data type of global items used to store job configs.\r\n   */\r\n  use(configDataType: string): void {\r\n    this.configDataType = configDataType;\r\n  }\r\n\r\n  /**\r\n   * Fetch global item containing job config of an App.\r\n   * @param appId ID of App\r\n   */\r\n  private async fetchJobConfigItem(\r\n    appId: string,\r\n  ): Promise<GlobalItem | undefined> {\r\n    return (\r\n      await this.context.query({\r\n        references: [appId],\r\n        data_types: [this.configDataType],\r\n      })\r\n    ).find(Boolean);\r\n  }\r\n\r\n  /**\r\n   * Get job config of an App.\r\n   * @param appId ID of App\r\n   */\r\n  async getJobConfig(appId: string): Promise<object | false> {\r\n    const item = await this.fetchJobConfigItem(appId);\r\n    if (item) {\r\n      return JSON.parse(item.value);\r\n    }\r\n    return false;\r\n  }\r\n\r\n  /**\r\n   * Get job config of an App.\r\n   * @param appId ID of App\r\n   * @param value Job config of an App\r\n   */\r\n  async createJobConfig(appId: string, value: object): Promise<object | false> {\r\n    const existingConfig = await this.fetchJobConfigItem(appId);\r\n    if (existingConfig) {\r\n      return false;\r\n    }\r\n    return await this.context.createValue(\r\n      value,\r\n      `Job config of App`,\r\n      undefined,\r\n      appId,\r\n      GlobalItemReferenceType.JOB,\r\n      this.configDataType,\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Delete job config of an App.\r\n   * @param appId ID of App\r\n   */\r\n  async deleteJobConfig(appId: string): Promise<boolean> {\r\n    const existingConfig = await this.fetchJobConfigItem(appId);\r\n    if (!existingConfig) {\r\n      return false;\r\n    }\r\n    await this.context.delete(existingConfig.id);\r\n    return true;\r\n  }\r\n}\r\n","import RationAI from './rationai';\r\nimport { GlobalItem } from './types/global-item';\r\nimport { GlobalStorageQuery } from './types/global-storage-query';\r\nimport { GlobalItemList } from './types/global-item-list';\r\nimport { PostGlobalItems } from './types/post-global-items';\r\nimport { GlobalItems } from './types/global-items';\r\nimport { GlobalItemReferenceType } from './types/global-item-reference-type';\r\nimport { PostGlobalItem } from './types/post-global-item';\r\nimport { GlobalDataCreatorType } from './types/global-data-creator-type';\r\nimport { PutGlobalItem } from './types/put-global-item';\r\nimport WsiMetadata from '../extensions/wsi-metadata';\r\nimport VisualizationTemplates from '../extensions/visualization-templates';\r\nimport AnnotPresets from '../extensions/annot-presets';\r\nimport JobConfig from '../extensions/job-config';\r\n\r\nexport default class GlobalStorage {\r\n  protected context: RationAI;\r\n  protected data: GlobalItem | null = null;\r\n\r\n  wsiMetadata: WsiMetadata;\r\n  visTemplates: VisualizationTemplates;\r\n  annotPresets: AnnotPresets;\r\n  jobConfig: JobConfig;\r\n\r\n  constructor(context: RationAI) {\r\n    this.context = context;\r\n    this.wsiMetadata = new WsiMetadata(this);\r\n    this.visTemplates = new VisualizationTemplates(this);\r\n    this.annotPresets = new AnnotPresets(this);\r\n    this.jobConfig = new JobConfig(this);\r\n  }\r\n\r\n  async get(itemId: string): Promise<GlobalItem> {\r\n    const globalItem: GlobalItem = await this.context.rawQuery(\r\n      `/global-storage/${itemId}`,\r\n    );\r\n    return globalItem;\r\n  }\r\n\r\n  async getValue(itemId: string): Promise<any> {\r\n    const item = await this.get(itemId);\r\n    if (item.type === 'string') {\r\n      try {\r\n        return JSON.parse(item.value);\r\n      } catch {\r\n        return item.value;\r\n      }\r\n    }\r\n    return item.value;\r\n  }\r\n\r\n  async query(query: GlobalStorageQuery): Promise<GlobalItem[]> {\r\n    const data: GlobalItemList = await this.context.rawQuery(\r\n      `/global-storage/query`,\r\n      {\r\n        method: 'PUT',\r\n        body: query,\r\n      },\r\n    );\r\n\r\n    return data.items;\r\n  }\r\n\r\n  async create(items: PostGlobalItems): Promise<GlobalItems> {\r\n    const createdItems: GlobalItems = await this.context.rawQuery(\r\n      `/global-storage`,\r\n      {\r\n        method: 'POST',\r\n        body: items,\r\n      },\r\n    );\r\n\r\n    return createdItems;\r\n  }\r\n\r\n  async createValue(\r\n    value: any,\r\n    name: string,\r\n    description?: string,\r\n    reference_id?: string,\r\n    reference_type?: GlobalItemReferenceType,\r\n    data_type?: string,\r\n  ): Promise<GlobalItem> {\r\n    value = JSON.stringify(value);\r\n\r\n    const newItem: PostGlobalItem = {\r\n      name: name,\r\n      description: description,\r\n      creator_id: this.context.userId,\r\n      creator_type: GlobalDataCreatorType.USER,\r\n      reference_id: reference_id,\r\n      reference_type: reference_type,\r\n      type: 'string',\r\n      value: value,\r\n      data_type: data_type,\r\n    };\r\n\r\n    return (await this.create(newItem)) as GlobalItem;\r\n  }\r\n\r\n  async update(itemId: string, item: PutGlobalItem): Promise<GlobalItem> {\r\n    const updatedItem: GlobalItem = await this.context.rawQuery(\r\n      `/global-storage/${itemId}`,\r\n      {\r\n        method: 'PUT',\r\n        body: item,\r\n      },\r\n    );\r\n    return updatedItem;\r\n  }\r\n\r\n  async updateValue(itemId: string, value: any): Promise<GlobalItem> {\r\n    const item: GlobalItem = await this.context.rawQuery(\r\n      `/global-storage/${itemId}`,\r\n    );\r\n    item.value = JSON.stringify(value);\r\n    const updatedItem: GlobalItem = await this.context.rawQuery(\r\n      `/global-storage/${itemId}`,\r\n      {\r\n        method: 'PUT',\r\n        body: item,\r\n      },\r\n    );\r\n    return updatedItem;\r\n  }\r\n\r\n  async delete(itemId: string): Promise<void> {\r\n    await this.context.rawQuery(`/global-storage/${itemId}`, {\r\n      method: 'DELETE',\r\n    });\r\n  }\r\n}\r\n","import { RawOptions } from '../../base';\r\nimport Root from '../root/root';\r\nimport GlobalStorage from './global-storage';\r\n\r\nexport default class RationAI {\r\n  static relativeApiPath = '/rationai';\r\n\r\n  context: Root;\r\n\r\n  //custom\r\n  globalStorage: GlobalStorage;\r\n\r\n  constructor(context: Root) {\r\n    this.context = context;\r\n    this.globalStorage = new GlobalStorage(this);\r\n  }\r\n\r\n  get userId(): string {\r\n    return this.context.userId;\r\n  }\r\n\r\n  async rawQuery(endpoint: string, options: RawOptions = {}): Promise<any> {\r\n    return this.context.rawQuery(\r\n      `${RationAI.relativeApiPath}${endpoint}`,\r\n      options,\r\n    );\r\n  }\r\n}\r\n","import { EmpationAPIOptions, RawAPI, RawOptions } from '../../base';\r\nimport { RootAPI } from '../../root';\r\nimport Scope from '../scope/scope';\r\nimport Apps from './apps';\r\nimport Cases from './cases';\r\nimport Examinations from './examinations';\r\nimport Slides from './slides';\r\nimport RationAI from '../rationai/rationai';\r\nimport { WorkbenchServiceApiV3CustomModelsExaminationsExamination } from './types/workbench-service-api-v-3-custom-models-examinations-examination';\r\n\r\nexport default class Root extends RootAPI {\r\n  static apiPath = '/v3';\r\n\r\n  //interface\r\n  protected raw: RawAPI;\r\n  protected defaultScopeKey: string = '';\r\n  rationai: RationAI;\r\n  version: string;\r\n  rootURI: string;\r\n\r\n  scopes: Map<string, Scope>;\r\n\r\n  //custom\r\n  apps: Apps;\r\n  cases: Cases;\r\n  examinations: Examinations;\r\n  slides: Slides;\r\n\r\n  constructor(options: EmpationAPIOptions) {\r\n    super(options);\r\n    this.version = 'v3';\r\n    this.rootURI = this.options.apiUrl + Root.apiPath;\r\n    this.raw = new RawAPI(this.rootURI);\r\n    this.rationai = new RationAI(this);\r\n\r\n    this.apps = new Apps(this);\r\n    this.cases = new Cases(this);\r\n    this.examinations = new Examinations(this);\r\n    this.slides = new Slides(this);\r\n\r\n    this.scopes = new Map<string, Scope>();\r\n  }\r\n\r\n  get defaultScope(): Scope | undefined {\r\n    return this.scopes.get(this.defaultScopeKey);\r\n  }\r\n\r\n  private async newScopeFrom(\r\n    examination: WorkbenchServiceApiV3CustomModelsExaminationsExamination,\r\n  ) {\r\n    const scope = new Scope(this);\r\n    await scope.from(examination);\r\n    this.scopes.set(examination.id, scope);\r\n    if (this.defaultScopeKey === '') this.defaultScopeKey = examination.id;\r\n    return scope;\r\n  }\r\n\r\n  private async newScopeUse(caseId: string, appId?: string) {\r\n    const scope = new Scope(this);\r\n    await scope.use(caseId, appId);\r\n    this.scopes.set(scope.activeExaminationId, scope);\r\n    if (this.defaultScopeKey === '')\r\n      this.defaultScopeKey = scope.activeExaminationId;\r\n    return scope;\r\n  }\r\n\r\n  async getScopeFrom(\r\n    examination: WorkbenchServiceApiV3CustomModelsExaminationsExamination,\r\n  ) {\r\n    return (\r\n      this.scopes.get(examination.id) || (await this.newScopeFrom(examination))\r\n    );\r\n  }\r\n\r\n  async getScopeUse(caseId: string, appId?: string) {\r\n    const matchingScopes = [...this.scopes.values()].filter(\r\n      (scp) =>\r\n        scp.activeCaseId === caseId &&\r\n        (appId ? scp.activeAppId === appId : true),\r\n    );\r\n    return matchingScopes.length > 0\r\n      ? matchingScopes[0]\r\n      : await this.newScopeUse(caseId, appId);\r\n  }\r\n\r\n  async rawQuery(endpoint: string, options: RawOptions = {}): Promise<any> {\r\n    await super.rawQuery(endpoint, options);\r\n    options = options || {};\r\n    options.headers = options.headers || {};\r\n    options.headers['User-Id'] = this.userId;\r\n    if (this.accessToken) {\r\n      options.headers['Authorization'] =\r\n        options.headers['Authorization'] || `Bearer ${this.rawToken}`;\r\n    }\r\n    return this.raw.http(endpoint, options);\r\n  }\r\n}\r\n"],"names":["__webpack_require__","exports","definition","key","o","Object","defineProperty","enumerable","get","obj","prop","prototype","hasOwnProperty","call","Symbol","toStringTag","value","EventSource","events","addOnceHandler","eventName","handler","userData","times","priority","self","this","count","onceHandler","event","removeHandler","addHandler","isFunction","index","length","handlers","Array","isArray","i","push","numberOfHandlers","removeAllHandlers","eventType","getHandler","apply","source","args","eventSource","getAwaitingHandler","Promise","resolve","loop","result","type","then","raiseEvent","eventArgs","raiseEventAwaiting","awaitingHandler","String","class2type","toString","STATUS_CODES","HTTPError","Error","code","message","extras","super","arguments","assign","name","suffix","replace","split","reduce","acc","c","charAt","toUpperCase","slice","httpErrorToName","statusCode","RawAPI","constructor","url","options","_parseQueryParams","params","undefined","k","v","URLSearchParams","_fetch","response","fetch","method","headers","body","responseType","e","status","statusText","error","ok","http","endpoint","hasBody","startsWith","query","JSON","stringify","AbstractAPI","getCallerName","orig","prepareStackTrace","stack","requires","getJwtTokenExpiresTimeout","token","exp","Date","now","parseJwtToken","parse","atob","sleep","ms","setTimeout","Logger","console","warn","info","debug","RootContext","RootAPI","accessToken","_tokenExpires","_rawToken","workbenchApiUrl","apiUrl","apiRootPath","endsWith","anonymousUserId","_userId","cached","from","withEvent","reset","tokenTimeout","userId","sub","use","defaultScopeKey","scopes","forEach","scp","clear","rawToken","rawQuery","eventObject","newToken","ScopeContext","ScopeAPI","Apps","context","data","_defaultApp","list","apps","tissues","stains","job_modes","app","items","name_short","vendor_name","getNumber","variable","Number","matchStringOnSeparatorGroup","str","separator","groupIdx","matches","RegExp","exec","CaseExplorer","customCases","caseHierarchy","caseTissues","caseStains","identifierSeparator","hierarchySpec","hierarchyNameOverrides","getCustomCases","map","caseObj","pathInHierarchy","getCaseHierarchyPath","pathFinished","prev","curr","val","getCaseValue","returnVal","getCase","caseId","find","cs","id","getCaseYear","getCaseMonth","getCaseDay","getCaseDescription","getCaseTissues","getCaseStains","isNaN","getCaseIdentifierPart","evaluateCaseValue","evalValue","caseValue","evaulateCaseYear","evaulateCaseMonth","evaulateCaseDay","evaluateCaseDescription","evaluateCaseTissues","evaluateCaseStains","evaulateCaseIdentifierPart","epochTime","created_at","getFullYear","getMonth","getDate","partIdx","parts","local_id","description","keys","stringToMatch","tokenString","tokens","filter","Boolean","searchTermRegex","join","match","matchStringOnTokens","every","tissue","includes","stain","hierarchyLevel","keyIdx","cases","currentHierarchyPath","levelName","lastLevel","groups","item","overrideName","hierarchy","search","filteredCases","localization","allTissues","entries","tisName","tisValue","locName","t","Map","values","allStains","stnName","stnValue","s","WsiExplorer","lastCaseId","slidesData","masksData","maskIdentifierSeparator","maskIdentifierValue","getAllSlides","slides","slide","masks","Cases","caseExplorer","wsiExplorer","Examinations","create","appId","case_id","app_id","skip","limit","examinationId","scope","Storage","getRaw","flush","content","set","valueRaw","dataRaw","erase","Annotations","upload","update","delete","externalIds","Collections","collectionId","collection","queryItems","createItems","deleteItem","itemId","Jobs","getJobs","jobId","Pixelmaps","pixelmapId","post","pixelmaps","getTile","level","tileX","tileY","uploadTile","tile","deleteTile","bulkGetTile","startX","startY","endX","endY","bulkUploadTile","tiles","Scope","scopeContext","_defaultExaminationId","_tokenRefetchInterval","activeExaminationId","activeCaseId","activeAppId","raw","apiPath","storage","annotations","collections","jobs","findExamination","examinations","creators","item_count","examination","ex","state","default","scopeToken","access_token","timeout","setInterval","clearInterval","scope_id","GlobalDataCreatorType","GlobalItemReferenceType","TemplateType","Slides","slideInfo","slideId","slideThumbnail","maxWidth","maxHeight","format","image_format","slideLabel","loadTile","x","y","WsiMetadata","defaultSlideMetadata","visualization","defaultMaskMetadata","getWsiMetadataItem","wsiId","isSlide","metadata","references","p","data_type","createWsiMetadataItem","createValue","WSI","getSlideMetadata","updateSlideMetadata","metadataItem","updatedItem","getShadersConfig","shaders","shadersConfig","shaderTemp","visTemplates","getTemplate","Shader","shaderTemplate","dataReferences","dataRefs","getVisualizations","slideVis","paramsTemplate","Params","background","Background","template","dataRef","visualizations","all","vis","Visualization","visTemplate","getMaskMetadata","maskId","templatesGlobalItemDataType","VisualizationTemplates","fetchTemplateItem","data_types","tmpl","createTemplate","deleteTemplate","existingTmpl","AnnotPresets","presetDataType","getPresetsItem","fresh","presetsItem","createPresetsItem","presets","getAnnotPresets","presetItem","lastModifiedAt","modified_at","mergePresets","primaryArr","secondaryArr","localVersion","newArr","secItem","some","primItem","createdAt","updateAnnotPresets","failOnParallelUpdate","remotePresetsItem","remotePresets","localPresets","successfulUpdate","retryAttempt","deleteAnnotPresets","JobConfig","configDataType","fetchJobConfigItem","getJobConfig","createJobConfig","JOB","deleteJobConfig","existingConfig","GlobalStorage","wsiMetadata","annotPresets","jobConfig","getValue","reference_id","reference_type","newItem","creator_id","creator_type","USER","updateValue","RationAI","globalStorage","relativeApiPath","Root","version","rootURI","rationai","defaultScope","newScopeFrom","newScopeUse","getScopeFrom","getScopeUse","matchingScopes"],"sourceRoot":""} \ No newline at end of file +var EmpationAPI;(()=>{var t={732:t=>{function e(t){return Promise.resolve().then((()=>{var e=new Error("Cannot find module '"+t+"'");throw e.code="MODULE_NOT_FOUND",e}))}e.keys=()=>[],e.resolve=e,e.id=732,t.exports=e}},e={};function i(n){var s=e[n];if(void 0!==s)return s.exports;var r=e[n]={exports:{}};return t[n](r,r.exports,i),r.exports}i.d=(t,e)=>{for(var n in e)i.o(e,n)&&!i.o(t,n)&&Object.defineProperty(t,n,{enumerable:!0,get:e[n]})},i.o=(t,e)=>Object.prototype.hasOwnProperty.call(t,e),i.r=t=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(t,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(t,"__esModule",{value:!0})};var n={};(()=>{"use strict";i.r(n),i.d(n,{AbstractAPI:()=>u,EventSource:()=>e,HTTPError:()=>o,Logger:()=>l,RawAPI:()=>a,RootAPI:()=>p,RootContext:()=>f,STATUS_CODES:()=>s,ScopeAPI:()=>y,ScopeContext:()=>v,V3:()=>t,getJwtTokenExpiresTimeout:()=>c,parseJwtToken:()=>d,sleep:()=>h});var t={};i.r(t),i.d(t,{Apps:()=>g,Cases:()=>C,Examinations:()=>_,RationAI:()=>rt,Root:()=>lt,Scope:()=>J,Slides:()=>B,Storage:()=>O});class e{constructor(){this.events={}}addOnceHandler(t,e,i,n,s){const r=this;n=n||1;let o=0;const a=function(i){return o++,o===n&&r.removeHandler(t,a),e(i)};this.addHandler(t,a,i,s)}addHandler(t,i,n=null,s=0){let r=this.events[t];if(r||(this.events[t]=r=[]),i&&e.isFunction(i)){let t=r.length,e={handler:i,userData:n||null,priority:s||0};for(r[t]=e;t>0&&r[t-1].priority{const r=i.length;!function o(a){if(a>=r||!i[a])return s("Resolved!"),null;n.eventSource=t,n.userData=i[a].userData;let u=i[a].handler(n);return u=u&&"promise"===e.type(u)?u:Promise.resolve(),u.then((()=>o(a+1)))}(0)}))}):null}raiseEvent(t,e){const i=this.getHandler(t);if(i)return i(this,e||{})}raiseEventAwaiting(t,e){const i=this.getAwaitingHandler(t);return i?i(this,e||{}):Promise.resolve("No handler for this event registered.")}static isFunction(t){return"function"===this.type(t)}static type(t){return null==t?String(t):this.class2type[t.toString()]||("function"==typeof t?"function":"object")}}e.class2type={"[object Boolean]":"boolean","[object Number]":"number","[object String]":"string","[object Function]":"function","[object AsyncFunction]":"function","[object Promise]":"promise","[object Array]":"array","[object Date]":"date","[object RegExp]":"regexp","[object Object]":"object"};const s={100:"Continue",101:"Switching protocols",102:"Processing",103:"Early Hints",200:"Ok",201:"Created",202:"Accepted",203:"Non Authoritative Information",204:"No Content",205:"Reset Content",206:"Partial Content",207:"Multi Status",300:"Multiple Choices",301:"Moved Permanently",302:"Moved Temporarily",303:"See Other",304:"Not Modified",305:"Use Proxy",307:"Temporary Redirect",308:"Permanent Redirect",400:"Bad Request",401:"Unauthorized",402:"Payment Required",403:"Forbidden",404:"Not Found",405:"Method Not Allowed",406:"Not Acceptable",407:"Proxy Authentication Required",408:"Request Timeout",409:"Conflict",410:"Gone",411:"Length Required",412:"Precondition Failed",413:"Request Too Long",414:"Request Uri Too Long",415:"Unsupported Media Type",416:"Requested Range Not Satisfiable",417:"Expectation Failed",418:"Im A Teapot",419:"Insufficient Space On Resource",420:"Method Failure",421:"Misdirected Request",422:"Unprocessable Entity",423:"Locked",424:"Failed Dependency",426:"Upgrade Required",428:"Precondition Required",429:"Too Many Requests",431:"Request Header Fields Too Large",451:"Unavailable For Legal Reasons",500:"Internal Server Error",501:"Not Implemented",502:"Bad Gateway",503:"Service Unavailable",504:"Gateway Timeout",505:"Http Version Not Supported",507:"Insufficient Storage",511:"Network Authentication Required"};var r=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class o extends Error{constructor(t,e,i){super(e||s[t]||`HTTP Code ${t}`),arguments.length>=3&&i&&Object.assign(this,i),this.name=function(t){const e=4==(t/100|0)||5==(t/100|0)?"error":"";return` ${String(s[t]||`HTTP Code ${t}`).replace(/error$/i,"")} ${e}`.split(" ").reduce(((t,e)=>t+(e?e.charAt(0).toUpperCase()+e.slice(1):"")))}(t),this.statusCode=t}}class a{constructor(t,e={}){this.url=t,this.options=e}_parseQueryParams(t){if(t){if("string"==typeof t)return t;if(t.constructor===Object||void 0===t.constructor)for(let e in t){null==t[e]&&delete t[e]}return`?${new URLSearchParams(t)}`}return""}_fetch(t,e){return r(this,void 0,void 0,(function*(){const i=yield fetch(t,{method:e.method,headers:e.headers,body:e.body});let n;try{n=yield i[e.responseType||"json"]()}catch(e){throw new o(500,`Failed to parse response data. Original status: ${i.status} | ${i.statusText}`,{url:t,error:e})}if(!i.ok)throw new o(i.status,i.statusText,{payload:n});return n}))}http(t,e){return r(this,void 0,void 0,(function*(){const i=!!e.body;return e.method=e.method||(i?"POST":"GET"),t.startsWith("/")||(t=`/${t}`),e.query=this._parseQueryParams(e.query),e.headers=e.headers||{},e.headers["Content-Type"]="application/json",e.body&&"string"!=typeof e.body?e.body=JSON.stringify(e.body):e.body=void 0,yield this._fetch(this.url+t+e.query,e)}))}}class u extends e{getCallerName(){const t=Error.prepareStackTrace;Error.prepareStackTrace=(t,e)=>e;const{stack:e}=new Error;Error.prepareStackTrace=t;return(null==e?void 0:e[2])||"unknown context"}requires(t,e){if(!e)throw`ArgumentError[${this.getCallerName()}] ${t} is missing - required property!`}}function c(t){return 1e3*t.exp-Date.now()||3e5}function d(t){return JSON.parse(atob(t.split(".")[1]))}function h(t){return new Promise((e=>setTimeout(e,t)))}class l{static error(...t){console.error("E:EmpationAPI",...t)}static warn(...t){console.warn("W:EmpationAPI",...t)}static info(...t){console.info("I:EmpationAPI",...t)}static debug(...t){console.debug("D:EmpationAPI",...t)}}class f{}class p extends u{constructor(t){if(super(),this.accessToken=null,this._tokenExpires=0,this._rawToken="",!t.workbenchApiUrl)throw"WB Api url is required!";let e;e=t.apiRootPath?t.apiRootPath.startsWith("/")?`${t.workbenchApiUrl}${t.apiRootPath}`:`${t.workbenchApiUrl}/${t.apiRootPath}`:t.workbenchApiUrl,e.endsWith("/")&&(e=e.slice(0,-1)),this.options={apiUrl:e,workbenchApiUrl:t.workbenchApiUrl,anonymousUserId:t.anonymousUserId||"anonymous",apiRootPath:t.apiRootPath||""},this._userId=this.options.anonymousUserId,this.cached={}}from(t,e=!0){if(!t)return this.reset();this._rawToken=t,e=e&&!this.accessToken,this.accessToken=d(t);const i=c(this.accessToken);this._tokenExpires=Date.now()+i/2;let n=this.accessToken.sub;if(!n)throw"Invalid User ID! Must be valid string shorter than 50 characters!";n.length>50&&(console.warn("User ID exceeded 50 characters! Using User ID shortened to first 50 characters!"),n=n.slice(0,50)),this.userId!==n&&(this._userId=n,e&&this.raiseEvent("init"))}use(t,e=!0){if(e=e&&!this._userId,this.reset(),!t||t.length>50)throw"Invalid User ID! Must be valid string shorter than 50 characters!";this._userId=t,e&&this.raiseEvent("init_no_token")}reset(){this._rawToken="",this._tokenExpires=0,this.accessToken=null,this._userId=this.options.anonymousUserId,this.defaultScopeKey="",this.scopes.forEach((t=>t.reset())),this.scopes.clear(),this.raiseEvent("reset")}get userId(){return this._userId}get rawToken(){return this._rawToken}rawQuery(t,e){return i=this,n=void 0,r=function*(){if(!this._userId)throw"User must be configured to access Empaia API: either provide a valid 'anonymous' user ID through env variables, or configure the Root API with a valid token.";if(this._tokenExpires>0&&Date.now()>this._tokenExpires){const t={newToken:""};yield this.raiseEventAwaiting("token-refresh",t),this.from(t.newToken)}},new((s=void 0)||(s=Promise))((function(t,e){function o(t){try{u(r.next(t))}catch(t){e(t)}}function a(t){try{u(r.throw(t))}catch(t){e(t)}}function u(e){var i;e.done?t(e.value):(i=e.value,i instanceof s?i:new s((function(t){t(i)}))).then(o,a)}u((r=r.apply(i,n||[])).next())}));var i,n,s,r}}class v{}class y extends u{}var m=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class g extends f{constructor(t){super(),this.data=null,this._defaultApp=null,this.context=t}list(){return m(this,void 0,void 0,(function*(){return this.data=yield this.context.rawQuery("/apps/query",{method:"PUT",body:{apps:null,tissues:null,stains:null,job_modes:null}})}))}query(t){return m(this,void 0,void 0,(function*(){return this.data=yield this.context.rawQuery("/apps/query",{method:"PUT",body:t})}))}default(){return m(this,void 0,void 0,(function*(){this.data||(yield this.list());for(let t of this.data.items)if("MAP3"===t.name_short&&"rationai"===t.vendor_name){this._defaultApp=t;break}if(!this._defaultApp)throw"Default APP not present in the infrastructure! Was it imported?";return this._defaultApp}))}}const x=t=>("string"==typeof t&&(t=Number(t)),t),w=(t,e,i,n)=>{const s=new RegExp(e).exec(t);return!(!s||i<1||i>=s.length)&&s[i]===n};var b=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class I{constructor(t,e){this.customCases=null,this.caseHierarchy=null,this.caseTissues=null,this.caseStains=null,this.identifierSeparator="",this.hierarchySpec=[],this.hierarchyNameOverrides={},this.context=t,this.integration=e}use(t,e,i={}){this.hierarchySpec=e,this.identifierSeparator=t,this.hierarchyNameOverrides=i}getCustomCases(){return b(this,void 0,void 0,(function*(){return this.customCases||(this.customCases=(yield this.context.list()).items.map((t=>Object.assign(Object.assign({},t),{pathInHierarchy:this.getCaseHierarchyPath(t)})))),this.customCases}))}getCaseHierarchyPath(t){if(!this.identifierSeparator||!this.hierarchySpec)throw"ArgumentError[CaseExplorer] identifierSeparator or hierarchySpec is missing - required property!";let e=!1;return this.hierarchySpec.reduce(((i,n)=>{const s=this.getCaseValue(n,t),r=e?i:`${i}/${s}`;return"OTHER"===s&&(e=!0),r}),"")}getCase(t){return b(this,void 0,void 0,(function*(){let e;return this.customCases&&(e=this.customCases.find((e=>e.id===t))),e||(e=yield this.context.get(t)),Object.assign(Object.assign({},e),{pathInHierarchy:this.getCaseHierarchyPath(e)})}))}getCaseValue(t,e){switch(t){case"year":return this.getCaseYear(e);case"month":return this.getCaseMonth(e);case"day":return this.getCaseDay(e);case"description":return this.getCaseDescription(e);case"tissues":return this.getCaseTissues(e);case"stains":return this.getCaseStains(e);default:if("id_part_"===t.slice(0,8)&&!isNaN(Number(t.slice(8))))return this.getCaseIdentifierPart(e,Number(t.slice(8)));throw`KeyError[CaseExplorer] "${t}" is not supported!`}}evaluateCaseValue(t,e,i){const n=this.getCaseValue(t,i);switch(t){case"year":return this.evaulateCaseYear(n,e);case"month":return this.evaulateCaseMonth(n,e);case"day":return this.evaulateCaseDay(n,e);case"description":return this.evaluateCaseDescription(n,e);case"tissues":return this.evaluateCaseTissues(n,e);case"stains":return this.evaluateCaseStains(n,e);default:return this.evaulateCaseIdentifierPart(n,e)}}getCaseYear(t){return(e=t.created_at,new Date(1e3*x(e)).getFullYear()).toString();var e}getCaseMonth(t){return(e=t.created_at,new Date(1e3*x(e)).getMonth()).toString();var e}getCaseDay(t){return(e=t.created_at,new Date(1e3*x(e)).getDate()).toString();var e}getCaseIdentifierPart(t,e){if(!this.identifierSeparator)throw"ArgumentError[CaseExplorer] identifierSeparator is missing - required property!";const i=new RegExp(this.identifierSeparator).exec(t.local_id||"");if(!i)return"OTHER";if(e<1||e>=i.length)throw'KeyError[CaseExplorer] invalid key "id_part_", group index is not valid!';return i[e]}getCaseDescription(t){return t.description||""}getCaseTissues(t){return Object.keys(t.tissues)}getCaseStains(t){return Object.keys(t.stains)}evaulateCaseYear(t,e){return t===e}evaulateCaseMonth(t,e){return t===e}evaulateCaseDay(t,e){return t===e}evaulateCaseIdentifierPart(t,e){return t===e}evaluateCaseDescription(t,e){return((t,e)=>{const i=e.split(" ").filter(Boolean).map((t=>`(?=.*\\b${t}\\b)`)),n=new RegExp(i.join(""),"gim");return null!==t.match(n)})(t,e)}evaluateCaseTissues(t,e){return e instanceof Array||(e=[e]),e.every((e=>t.includes(e)))}evaluateCaseStains(t,e){return e instanceof Array||(e=[e]),e.every((e=>t.includes(e)))}hierarchyLevel(t,e,i,n,s,r,o){return b(this,void 0,void 0,(function*(){if(e>=t.length)return{levelId:s,levelName:r,lastLevel:!0,parent:o,items:i.map((t=>Object.assign(Object.assign({},t),{pathInHierarchy:n})))};const a=(u=i=>{const n=this.getCaseValue(t[e],i);return Array.isArray(n)?n[0]||"":n},i.reduce(((t,e)=>{var i;return(t[i=u(e)]||(t[i]=[])).push(e),t}),{}));var u;const c={levelName:r,levelId:s,lastLevel:!1,items:[]};return c.items=yield Promise.all(Object.keys(a).map((i=>b(this,void 0,void 0,(function*(){var s;let r=(null===(s=this.hierarchyNameOverrides[t[e]])||void 0===s?void 0:s[i])||(yield this.integration.translatePathSpec(t[e],i))||i;if("OTHER"===i){const e=yield this.hierarchyLevel(t,t.length,a[i],`${n}/${r}`,i,r);return e.parent=c,e}const o=yield this.hierarchyLevel(t,e+1,a[i],`${n}/${r}`,i,r);return o.parent=c,o}))))),c}))}hierarchy(){return b(this,void 0,void 0,(function*(){if(!this.caseHierarchy){const t=yield this.getCustomCases();this.caseHierarchy=yield this.hierarchyLevel(this.hierarchySpec,0,t,"")}return this.caseHierarchy}))}search(t){return b(this,void 0,void 0,(function*(){let e=yield this.getCustomCases();return t.forEach((({key:t,value:i})=>e=e.filter((e=>this.evaluateCaseValue(t,i,e))))),e}))}tissues(t="EN"){return b(this,void 0,void 0,(function*(){if(!this.caseTissues){const e=yield this.getCustomCases(),i=[];e.forEach((e=>Object.entries(e.tissues).map((([e,i])=>({name:e,locName:i[t]}))).forEach((t=>i.push(t))))),this.caseTissues=[...new Map(i.map((t=>[JSON.stringify([t.name,t.locName]),t]))).values()]}return this.caseTissues}))}stains(t="EN"){return b(this,void 0,void 0,(function*(){if(!this.caseStains){const e=yield this.getCustomCases(),i=[];e.forEach((e=>Object.entries(e.stains).map((([e,i])=>({name:e,locName:i[t]}))).forEach((t=>i.push(t))))),this.caseStains=[...new Map(i.map((t=>[JSON.stringify([t.name,t.locName]),t]))).values()]}return this.caseStains}))}}var T=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class P{constructor(t,e){this.lastCaseId=null,this.data=null,this.slidesData=null,this.masksData=null,this.maskIdentifierSeparator="",this.maskIdentifierValue="",this.context=t,this.integration=e}use(t,e){this.maskIdentifierSeparator=t,this.maskIdentifierValue=e}getAllSlides(t){return T(this,void 0,void 0,(function*(){return this.lastCaseId===t&&this.data||(this.data=(yield this.context.slides(t)).items),this.data}))}slides(t){return T(this,void 0,void 0,(function*(){return this.lastCaseId===t&&this.slidesData||(this.slidesData=(yield this.getAllSlides(t)).filter((t=>!w(t.local_id||"",this.maskIdentifierSeparator,1,this.maskIdentifierValue)))),this.slidesData}))}masks(t){return T(this,void 0,void 0,(function*(){return this.lastCaseId===t&&this.masksData||(this.masksData=(yield this.getAllSlides(t)).filter((t=>w(t.local_id||"",this.maskIdentifierSeparator,1,this.maskIdentifierValue)))),this.masksData}))}}var S=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class C extends f{constructor(t){super(),this.data=null,this.context=t,this.caseExplorer=new I(this,t.integration),this.wsiExplorer=new P(this,t.integration)}list(){return S(this,void 0,void 0,(function*(){return this.data||(this.data=yield this.context.rawQuery("/cases")),this.data}))}get(t){return S(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/cases/${t}`)}))}slides(t){return S(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/cases/${t}/slides`)}))}}var E=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class _ extends f{constructor(t){super(),this.data=null,this.context=t}create(t,e){return E(this,void 0,void 0,(function*(){const i=this.context;return i.requires("caseId",t),i.rawQuery("/examinations",{method:"PUT",body:{case_id:t,app_id:e}})}))}query(t,e,i){return E(this,void 0,void 0,(function*(){return this.context.rawQuery("/examinations/query",{method:"PUT",body:t,query:{skip:e,limit:i}})}))}get(t){return E(this,void 0,void 0,(function*(){const e=this.context;return e.requires("examinationId",t),e.rawQuery(`/examinations/${t}`)}))}scope(t){return E(this,void 0,void 0,(function*(){const e=this.context;return e.requires("examinationId",t),e.rawQuery(`/examinations/${t}/scope`,{method:"PUT"})}))}}var $=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class O extends v{constructor(t){super(),this.data=null,this.context=t}getRaw(){return $(this,void 0,void 0,(function*(){return this.data||(this.data=yield this.context.rawQuery("/app-ui-storage/user")),this.data}))}flush(){return $(this,void 0,void 0,(function*(){return this.data?yield this.context.rawQuery("/app-ui-storage/user",{method:"PUT",body:this.data}):null}))}get(t){return $(this,void 0,void 0,(function*(){const e=yield this.getRaw();return"string"==typeof e.content[t]?JSON.parse(e.content[t]):e.content[t]}))}set(t,e,i){return $(this,void 0,void 0,(function*(){const n=JSON.stringify(e),s=yield this.getRaw();s.content[t]=n,this.data=s,i&&this.flush()}))}erase(){return $(this,void 0,void 0,(function*(){this.data={content:{}},this.flush()}))}}var k=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class A extends v{constructor(t){super(),this.context=t}query(t,e=!0){return k(this,void 0,void 0,(function*(){return yield this.context.rawQuery("/annotations/query",{method:"PUT",query:{with_classes:e},body:t})}))}get(t,e=!0){return k(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/annotations/${t}`,{query:{with_classes:e}})}))}createMany(t,e={}){return k(this,void 0,void 0,(function*(){return yield this.context.rawQuery("/annotations",{method:"POST",query:e,body:t})}))}create(t,e={}){return k(this,void 0,void 0,(function*(){return yield this.context.rawQuery("/annotations",{method:"POST",query:e,body:t})}))}deleteById(t){return k(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/annotations/${t}`,{method:"DELETE"})}))}delete(t){return k(this,void 0,void 0,(function*(){if(!t.id)throw"Cannot delete annotation without ID property!";const e=yield this.deleteById(t.id);if(t.classes)for(let e of t.classes)e.id&&(yield this.deleteClass(e.id));return e}))}update(t,e,i={}){return k(this,void 0,void 0,(function*(){return yield this.deleteById(t),!i.externalIds&&e.id&&(i.externalIds=!0),yield this.create(e,i)}))}addClass(t){return k(this,void 0,void 0,(function*(){return yield this.context.rawQuery("/classes",{method:"POST",body:t})}))}addClassMany(t){return k(this,void 0,void 0,(function*(){return yield this.context.rawQuery("/classes",{method:"POST",body:t})}))}getClass(t){return k(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/classes/${t}`)}))}deleteClass(t){return k(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/classes/${t}`,{method:"DELETE"})}))}queryClasses(t){return k(this,void 0,void 0,(function*(){return yield this.context.rawQuery("/classes/query",{method:"PUT",body:t})}))}}var U=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class N extends v{constructor(t){super(),this.data=null,this.context=t}get(t){return U(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/collections/${t}`)}))}create(t){return U(this,void 0,void 0,(function*(){return yield this.context.rawQuery("/collections",{method:"POST",body:t})}))}delete(t){return U(this,void 0,void 0,(function*(){yield this.context.rawQuery(`/collections/${t}`,{method:"DELETE"})}))}queryItems(t,e){return U(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/collections/${t}/items/query`,{method:"PUT",body:e})}))}createItems(t,e){return U(this,void 0,void 0,(function*(){yield this.context.rawQuery(`/collections/${t}/items`,{method:"POST",body:Object.assign({},e)})}))}deleteItem(t,e){return U(this,void 0,void 0,(function*(){yield this.context.rawQuery(`/collections/${t}/items/${e}`,{method:"DELETE"})}))}}var j=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class D extends v{constructor(t){super(),this.data=null,this.context=t}getJobs(){return j(this,void 0,void 0,(function*(){return(yield this.context.rawQuery("/jobs")).items}))}get(t){return j(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/jobs/${t}`)}))}}var R=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class Q extends v{constructor(t){super(),this.data=null,this.context=t}get(t){return R(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/pixelmaps/${t}`)}))}post(t){return R(this,void 0,void 0,(function*(){return yield this.context.rawQuery("/pixelmaps",{method:"POST",body:t})}))}delete(t){return R(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/pixelmaps/${t}`,{method:"DELETE"})}))}query(t){return R(this,void 0,void 0,(function*(){return(yield this.context.rawQuery("/pixelmaps/query",{method:"PUT",body:t})).items}))}getTile(t,e,i,n){return R(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/pixelmaps/${t}/level/${e}/position/${i}/${n}/data`,{responseType:"blob"})}))}uploadTile(t,e,i,n,s){return R(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/pixelmaps/${t}/level/${e}/position/${i}/${n}/data`,{method:"PUT",body:s})}))}deleteTile(t,e,i,n){return R(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/pixelmaps/${t}/level/${e}/position/${i}/${n}/data`,{method:"DELETE"})}))}bulkGetTile(t,e,i,n,s,r){return R(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/pixelmaps/${t}/level/${e}/position/start/${i}/${n}/end/${s}/${r}/data`,{responseType:"blob"})}))}bulkUploadTile(t,e,i,n,s,r,o){return R(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/pixelmaps/${t}/level/${e}/position/start/${i}/${n}/end/${s}/${r}/data`,{method:"PUT",body:o})}))}}var q=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class M extends y{constructor(t){super(),this.scopeContext=null,this._defaultExaminationId="",this._tokenRefetchInterval=null,this.activeExaminationId="",this.activeCaseId="",this.activeAppId="",this.context=t,this.raw=new a(this.context.options.apiUrl+M.apiPath),this.storage=new O(this),this.annotations=new A(this),this.collections=new N(this),this.jobs=new D(this),this.pixelmaps=new Q(this)}use(t,e=void 0){return q(this,void 0,void 0,(function*(){this.requires("root::userId",this.context.userId);const i=e=>q(this,void 0,void 0,(function*(){let i=yield this.context.examinations.query({apps:[e],creators:[this.context.userId]});if(i.item_count>0){let t=i.items.find((t=>"OPEN"===t.state));if(t)return t}return yield this.context.examinations.create(t,e)}));let n;if(e?n=yield i(e):this._defaultExaminationId&&(n=yield this.context.examinations.get(this._defaultExaminationId)),!n){let t=yield this.context.apps.default();n=yield i(t.app_id),this._defaultExaminationId=n.id}yield this.from(n)}))}get scopeToken(){var t;return(null===(t=this.scopeContext)||void 0===t?void 0:t.access_token)||""}get id(){var t;return(null===(t=this.scopeContext)||void 0===t?void 0:t.scope_id)||""}from(t){return q(this,void 0,void 0,(function*(){this.reset(),this.scopeContext=yield this.context.examinations.scope(t.id),this.activeCaseId=t.case_id,this.activeAppId=t.app_id,this.activeExaminationId=t.id;const e=c(d(this.scopeContext.access_token));this._tokenRefetchInterval=setInterval((()=>q(this,void 0,void 0,(function*(){this.scopeContext=yield this.context.examinations.scope(t.id)}))),e),this.raiseEvent("init")}))}reset(){this.activeExaminationId="",this.scopeContext=null,this._tokenRefetchInterval&&(clearInterval(this._tokenRefetchInterval),this._tokenRefetchInterval=null,this.raiseEvent("reset"))}rawQuery(t,e){var i,n;return q(this,void 0,void 0,(function*(){this.requires("this.scopeContext",this.scopeContext),(e=e||{}).headers=e.headers||{},e.headers.Authorization=`Bearer ${null===(i=this.scopeContext)||void 0===i?void 0:i.access_token}`,t&&!t.startsWith("/")&&(t=`/${t}`);try{return yield this.raw.http(`/${null===(n=this.scopeContext)||void 0===n?void 0:n.scope_id}${t}`,e)}catch(i){if(401===i.statusCode)return this.scopeContext=yield this.context.examinations.scope(this.activeExaminationId),yield this.raw.http(`/${this.scopeContext.scope_id}${t}`,e);throw i}}))}}M.apiPath="/v3/scopes";const J=M;var H,L,V,F=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class B extends f{constructor(t){super(),this.data=null,this.context=t}slideInfo(t){return F(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/slides/${t}/info`)}))}slideThumbnail(t,e,i,n){return F(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/slides/${t}/thumbnail/max_size/${e}/${i}`,{query:{image_format:n},responseType:"blob"})}))}slideLabel(t,e,i,n){return F(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/slides/${t}/label/max_size/${e}/${i}`,{query:{image_format:n},responseType:"blob"})}))}loadTile(t,e,i,n,s){return F(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/slides/${t}/tile/level/${e}/tile/${i}/${n}`,{query:{image_format:s},responseType:"blob"})}))}}!function(t){t.USER="user",t.SCOPE="scope",t.JOB="job"}(H||(H={})),function(t){t.ANNOTATION="annotation",t.COLLECTION="collection",t.CLASS="class",t.PRIMITIVE="primitive",t.WSI="wsi",t.CASE="case",t.USER="user",t.SCOPE="scope",t.JOB="job"}(L||(L={})),function(t){t.Background="background",t.Params="params",t.Shader="shader",t.Visualization="visualization"}(V||(V={}));var z=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class W{constructor(t){this.data=null,this.defaultSlideMetadata={visualization:{}},this.defaultMaskMetadata={},this.context=t}getWsiMetadataItem(t,e=!0){return z(this,void 0,void 0,(function*(){let i=(yield this.context.query({references:[t]})).find((t=>t.data_type===(e?"slide":"mask")+"_metadata"));return i||(i=yield this.createWsiMetadataItem(t,e?this.defaultSlideMetadata:this.defaultMaskMetadata)),i}))}createWsiMetadataItem(t,e,i=!0){return z(this,void 0,void 0,(function*(){return yield this.context.createValue(e,`Metadata of ${i?"slide":"mask"} ${t}`,void 0,t,L.WSI,(i?"slide":"mask")+"_metadata")}))}getSlideMetadata(t){return z(this,void 0,void 0,(function*(){return JSON.parse((yield this.getWsiMetadataItem(t)).value)}))}updateSlideMetadata(t,e){return z(this,void 0,void 0,(function*(){const i=yield this.getWsiMetadataItem(t);try{const t=yield this.context.update(i.id,Object.assign(Object.assign({},i),{value:JSON.stringify(e)}));return JSON.parse(t.value)}catch(t){return!1}}))}getShadersConfig(t){return z(this,void 0,void 0,(function*(){const e={};for(let i=0;iz(this,void 0,void 0,(function*(){return Object.assign(Object.assign({},yield this.context.visTemplates.getTemplate(V.Visualization,t.visTemplate)),{name:t.name,shaders:yield this.getShadersConfig(t.shaders)})}))))))}}))}getMaskMetadata(t){return z(this,void 0,void 0,(function*(){return JSON.parse((yield this.getWsiMetadataItem(t,!1)).value)}))}}var G=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};const K="vis_templates";class Y{constructor(t){this.data=null,this.context=t}fetchTemplateItem(t,e){return G(this,void 0,void 0,(function*(){return(yield this.context.query({references:[null],data_types:[`${K}_${t}`]})).find((t=>t.name===e))}))}getTemplate(t,e){return G(this,void 0,void 0,(function*(){const i=yield this.fetchTemplateItem(t,e);return!!i&&JSON.parse(i.value)}))}createTemplate(t,e,i){return G(this,void 0,void 0,(function*(){return!(yield this.fetchTemplateItem(t,e))&&(yield this.context.createValue(i,`${e}`,void 0,void 0,void 0,`${K}_${t}`))}))}deleteTemplate(t,e){return G(this,void 0,void 0,(function*(){const i=yield this.fetchTemplateItem(t,e);return!!i&&(yield this.context.delete(i.id),!0)}))}}var X=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class Z{constructor(t){this.data=null,this.presetDataType="annot_presets",this.context=t}use(t){this.presetDataType=t}getPresetsItem(t=!1){return X(this,void 0,void 0,(function*(){if(!this.data||t){let t=(yield this.context.query({references:[null],data_types:[this.presetDataType]})).find((t=>!0));t||(t=yield this.createPresetsItem({presets:[]})),this.data=t}return this.data}))}createPresetsItem(t){return X(this,void 0,void 0,(function*(){return yield this.context.createValue(t,"Global annotation presets",void 0,void 0,void 0,this.presetDataType)}))}getAnnotPresets(t=!1){return X(this,void 0,void 0,(function*(){const e=yield this.getPresetsItem(t);return{presets:JSON.parse(e.value).presets,lastModifiedAt:e.modified_at}}))}mergePresets(t,e,i){const n=[...t];return e.forEach((t=>n.some((e=>e.id===t.id))||!t.createdAt||t.createdAt<=i?null:n.push(t))),n}updateAnnotPresets(t,e,i=!1){return X(this,void 0,void 0,(function*(){const n=yield this.getPresetsItem(!0),s=JSON.parse(n.value).presets;let r=t,o=!0;if(n.modified_at!==e){if(i)return{presets:s,successfulUpdate:!1,lastModifiedAt:n.modified_at};r=this.mergePresets(s,r,e),o=!1}try{const t=yield this.context.update(n.id,Object.assign(Object.assign({},n),{value:JSON.stringify({presets:r})}));return{presets:JSON.parse(t.value).presets,successfulUpdate:o,lastModifiedAt:t.modified_at}}catch(t){if(409===t.statusCode){const t=yield this.updateAnnotPresets(r,n.modified_at);return Object.assign(Object.assign({},t),{successfulUpdate:o})}throw t}}))}deleteAnnotPresets(){return X(this,void 0,void 0,(function*(){const t=yield this.getPresetsItem(!0);yield this.context.delete(t.id),this.data=null}))}}var tt=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class et{constructor(t){this.data=null,this.configDataType="app_job_config",this.context=t}use(t){this.configDataType=t}fetchJobConfigItem(t){return tt(this,void 0,void 0,(function*(){return(yield this.context.query({references:[t],data_types:[this.configDataType]})).find(Boolean)}))}getJobConfig(t){return tt(this,void 0,void 0,(function*(){const e=yield this.fetchJobConfigItem(t);return!!e&&JSON.parse(e.value)}))}createJobConfig(t,e){return tt(this,void 0,void 0,(function*(){return!(yield this.fetchJobConfigItem(t))&&(yield this.context.createValue(e,"Job config of App",void 0,t,L.JOB,this.configDataType))}))}deleteJobConfig(t){return tt(this,void 0,void 0,(function*(){const e=yield this.fetchJobConfigItem(t);return!!e&&(yield this.context.delete(e.id),!0)}))}}var it=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class nt{constructor(t){this.data=null,this.context=t,this.wsiMetadata=new W(this),this.visTemplates=new Y(this),this.annotPresets=new Z(this),this.jobConfig=new et(this)}get(t){return it(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/global-storage/${t}`)}))}getValue(t){return it(this,void 0,void 0,(function*(){const e=yield this.get(t);if("string"===e.type)try{return JSON.parse(e.value)}catch(t){return e.value}return e.value}))}query(t){return it(this,void 0,void 0,(function*(){return(yield this.context.rawQuery("/global-storage/query",{method:"PUT",body:t})).items}))}create(t){return it(this,void 0,void 0,(function*(){return yield this.context.rawQuery("/global-storage",{method:"POST",body:t})}))}createValue(t,e,i,n,s,r){return it(this,void 0,void 0,(function*(){t=JSON.stringify(t);const o={name:e,description:i,creator_id:this.context.userId,creator_type:H.USER,reference_id:n,reference_type:s,type:"string",value:t,data_type:r};return yield this.create(o)}))}update(t,e){return it(this,void 0,void 0,(function*(){return yield this.context.rawQuery(`/global-storage/${t}`,{method:"PUT",body:e})}))}updateValue(t,e){return it(this,void 0,void 0,(function*(){const i=yield this.context.rawQuery(`/global-storage/${t}`);return i.value=JSON.stringify(e),yield this.context.rawQuery(`/global-storage/${t}`,{method:"PUT",body:i})}))}delete(t){return it(this,void 0,void 0,(function*(){yield this.context.rawQuery(`/global-storage/${t}`,{method:"DELETE"})}))}}class st{constructor(t){this.context=t,this.globalStorage=new nt(this)}get userId(){return this.context.userId}rawQuery(t,e={}){return i=this,n=void 0,r=function*(){return this.context.rawQuery(`${st.relativeApiPath}${t}`,e)},new((s=void 0)||(s=Promise))((function(t,e){function o(t){try{u(r.next(t))}catch(t){e(t)}}function a(t){try{u(r.throw(t))}catch(t){e(t)}}function u(e){var i;e.done?t(e.value):(i=e.value,i instanceof s?i:new s((function(t){t(i)}))).then(o,a)}u((r=r.apply(i,n||[])).next())}));var i,n,s,r}}st.relativeApiPath="/rationai";const rt=st;class ot{constructor(t){if(void 0===t._context)throw Error("Private property not configured.");this.context=t._context,this.props=t}translatePathSpec(t,e){return i=this,n=void 0,r=function*(){return e},new((s=void 0)||(s=Promise))((function(t,e){function o(t){try{u(r.next(t))}catch(t){e(t)}}function a(t){try{u(r.throw(t))}catch(t){e(t)}}function u(e){var i;e.done?t(e.value):(i=e.value,i instanceof s?i:new s((function(t){t(i)}))).then(o,a)}u((r=r.apply(i,n||[])).next())}));var i,n,s,r}}var at=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class ut{constructor(){throw Error("Not instantiable.")}static register(t,e){return this.items.set(t,e),e}static get(t){return this.items.get(t)}}ut.items=new Map;const ct=ut;ct.register("default",ot),ct.register("lsaai",class extends ot{translatePathSpec(t,e){return at(this,void 0,void 0,(function*(){return this.context.accessToken&&this.props.userinfo?(this.userData||(yield this.parseUserInstitutionsAndProjects()),"id_part_1"===t?this.userData.institutions[e]||e:"id_part_2"===t&&this.userData.projects[e]||e):e}))}parseGroupName(t){if(!t)return{name:t,slug:t};const e=(t=decodeURIComponent(t).trim()).match(/^(.+)\.\s*([^.\s]{1,5})$/);return e&&3===e.length?{name:e[1].trim(),slug:e[2].trim()}:t.length>5?(console.warn("Invalid group name",t),{}):{name:t,slug:t}}parseUserInstitutionsAndProjects(){var t;return at(this,void 0,void 0,(function*(){const e=yield fetch(this.props.userinfo,{headers:{Authorization:`Bearer ${this.context.accessToken}`}}),i=yield e.text();if(!e.ok)throw new Error(`Failed to fetch user info! ${e.statusText}. ${i}`);const n=JSON.parse(i),s=this.userData={projects:{"":{label:"NO PROJECT",value:"",rights:["write"]}},institutions:{"":{label:"PUBLIC DATA",value:""}}};try{const e=n.eduperson_entitlement;if(e)for(const i of e){const e=i.match(/^.*?:group:ration_ai:([^#\n]*)/);if(e&&e.length>1){const i=null===(t=e[1])||void 0===t?void 0:t.split(":");if(i){const t=i.length,e=this.parseGroupName(i[0]),n=this.parseGroupName(t>1?i[1]:""),r=this.parseGroupName(t>2?i[2]:null);n.slug&&!s.institutions[n.slug]&&(s.institutions[n.slug]={label:n.name,value:n.slug}),e.slug&&!s.projects[e.slug]?s.projects[e.slug]={label:e.name,value:e.slug,rights:r?[r]:[]}:r&&e.slug&&s.projects[e.slug].rights.push(r.slug)}}else console.warn("Ignored entitlement",i)}else console.warn("User info data does not contain access information! Is OAUTH scope set correctly?")}catch(t){console.warn("Could not decide user authorization capabilities!",t)}}))}});var dt=function(t,e,i,n){return new(i||(i=Promise))((function(s,r){function o(t){try{u(n.next(t))}catch(t){r(t)}}function a(t){try{u(n.throw(t))}catch(t){r(t)}}function u(t){var e;t.done?s(t.value):(e=t.value,e instanceof i?e:new i((function(t){t(e)}))).then(o,a)}u((n=n.apply(t,e||[])).next())}))};class ht extends p{constructor(t){super(t),this.defaultScopeKey="",this.version="v3",this.rootURI=this.options.apiUrl+ht.apiPath,this.raw=new a(this.rootURI),t.integrationOptions||(t.integrationOptions={}),t.integrationOptions._context=this;const e=t.integrationOptions.implementation||"default",i=ct.get(e);if(!i)throw new Error(`Could not instantiate integration provider ${e} - is it a valid name?`);this.integration=new i(t.integrationOptions),this.apps=new g(this),this.cases=new C(this),this.examinations=new _(this),this.slides=new B(this),this.rationai=new rt(this),this.scopes=new Map}get defaultScope(){return this.scopes.get(this.defaultScopeKey)}newScopeFrom(t){return dt(this,void 0,void 0,(function*(){const e=new J(this);return yield e.from(t),this.scopes.set(t.id,e),""===this.defaultScopeKey&&(this.defaultScopeKey=t.id),e}))}newScopeUse(t,e){return dt(this,void 0,void 0,(function*(){const i=new J(this);return yield i.use(t,e),this.scopes.set(i.activeExaminationId,i),""===this.defaultScopeKey&&(this.defaultScopeKey=i.activeExaminationId),i}))}getScopeFrom(t){return dt(this,void 0,void 0,(function*(){return this.scopes.get(t.id)||(yield this.newScopeFrom(t))}))}getScopeUse(t,e){return dt(this,void 0,void 0,(function*(){const i=[...this.scopes.values()].filter((i=>i.activeCaseId===t&&(!e||i.activeAppId===e)));return i.length>0?i[0]:yield this.newScopeUse(t,e)}))}rawQuery(t,e={}){const i=Object.create(null,{rawQuery:{get:()=>super.rawQuery}});return dt(this,void 0,void 0,(function*(){return yield i.rawQuery.call(this,t,e),(e=e||{}).headers=e.headers||{},e.headers["User-Id"]=this.userId,this.accessToken&&(e.headers.Authorization=e.headers.Authorization||`Bearer ${this.rawToken}`),this.raw.http(t,e)}))}}ht.apiPath="/v3";const lt=ht})(),EmpationAPI=n})(); +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"empationapi.js","mappings":"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","sources":["webpack://EmpationAPI/./src/v3/extensions/ lazy namespace object","webpack://EmpationAPI/webpack/bootstrap","webpack://EmpationAPI/webpack/runtime/define property getters","webpack://EmpationAPI/webpack/runtime/hasOwnProperty shorthand","webpack://EmpationAPI/webpack/runtime/make namespace object","webpack://EmpationAPI/./src/events.ts","webpack://EmpationAPI/./src/status-codes.ts","webpack://EmpationAPI/./src/base.ts","webpack://EmpationAPI/./src/utils.ts","webpack://EmpationAPI/./src/root.ts","webpack://EmpationAPI/./src/scope.ts","webpack://EmpationAPI/./src/v3/root/apps.ts","webpack://EmpationAPI/./src/v3/extensions/utils.ts","webpack://EmpationAPI/./src/v3/extensions/case-explorer.ts","webpack://EmpationAPI/./src/v3/extensions/wsi-explorer.ts","webpack://EmpationAPI/./src/v3/root/cases.ts","webpack://EmpationAPI/./src/v3/root/examinations.ts","webpack://EmpationAPI/./src/v3/scope/storage.ts","webpack://EmpationAPI/./src/v3/scope/annotations.ts","webpack://EmpationAPI/./src/v3/scope/collections.ts","webpack://EmpationAPI/./src/v3/scope/jobs.ts","webpack://EmpationAPI/./src/v3/scope/pixelmaps.ts","webpack://EmpationAPI/./src/v3/scope/scope.ts","webpack://EmpationAPI/./src/v3/rationai/types/global-data-creator-type.ts","webpack://EmpationAPI/./src/v3/rationai/types/global-item-reference-type.ts","webpack://EmpationAPI/./src/v3/extensions/types/template-type.ts","webpack://EmpationAPI/./src/v3/root/slides.ts","webpack://EmpationAPI/./src/v3/extensions/wsi-metadata.ts","webpack://EmpationAPI/./src/v3/extensions/visualization-templates.ts","webpack://EmpationAPI/./src/v3/extensions/annot-presets.ts","webpack://EmpationAPI/./src/v3/extensions/job-config.ts","webpack://EmpationAPI/./src/v3/rationai/global-storage.ts","webpack://EmpationAPI/./src/v3/rationai/rationai.ts","webpack://EmpationAPI/./src/v3/integration/default.ts","webpack://EmpationAPI/./src/v3/integration/integration-manager.ts","webpack://EmpationAPI/./src/v3/integration/index.ts","webpack://EmpationAPI/./src/v3/integration/lsaai.ts","webpack://EmpationAPI/./src/v3/root/root.ts"],"sourcesContent":["function webpackEmptyAsyncContext(req) {\n\t// Here Promise.resolve().then() is used instead of new Promise() to prevent\n\t// uncaught exception popping up in devtools\n\treturn Promise.resolve().then(() => {\n\t\tvar e = new Error(\"Cannot find module '\" + req + \"'\");\n\t\te.code = 'MODULE_NOT_FOUND';\n\t\tthrow e;\n\t});\n}\nwebpackEmptyAsyncContext.keys = () => ([]);\nwebpackEmptyAsyncContext.resolve = webpackEmptyAsyncContext;\nwebpackEmptyAsyncContext.id = 732;\nmodule.exports = webpackEmptyAsyncContext;","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","/**\r\n * OpenSeadragon inspired event system for reduction of dependencies and uniform event approach\r\n */\r\n\r\nexport type EventHandler = (event: any) => void;\r\n\r\nexport class EventSource {\r\n  events: { [key: string]: any } = {};\r\n\r\n  /**\r\n   * Add an event handler to be triggered only once (or a given number of times)\r\n   * for a given event. It is not removable with removeHandler().\r\n   * @function\r\n   * @param {String} eventName - Name of event to register.\r\n   * @param {EventHandler} handler - Function to call when event\r\n   * is triggered.\r\n   * @param {Object} [userData=null] - Arbitrary object to be passed unchanged\r\n   * to the handler.\r\n   * @param {Number} [times=1] - The number of times to handle the event\r\n   * before removing it.\r\n   * @param {Number} [priority=0] - Handler priority. By default, all priorities are 0. Higher number = priority.\r\n   */\r\n  addOnceHandler(\r\n    eventName: string,\r\n    handler: EventHandler,\r\n    userData: object,\r\n    times: number,\r\n    priority: number,\r\n  ) {\r\n    const self = this;\r\n    times = times || 1;\r\n    let count = 0;\r\n    const onceHandler = function (event: object) {\r\n      count++;\r\n      if (count === times) {\r\n        self.removeHandler(eventName, onceHandler);\r\n      }\r\n      return handler(event);\r\n    };\r\n    this.addHandler(eventName, onceHandler, userData, priority);\r\n  }\r\n\r\n  /**\r\n   * Add an event handler for a given event.\r\n   * @function\r\n   * @param {String} eventName - Name of event to register.\r\n   * @param {EventHandler} handler - Function to call when event is triggered.\r\n   * @param {Object} [userData=null] - Arbitrary object to be passed unchanged to the handler.\r\n   * @param {Number} [priority=0] - Handler priority. By default, all priorities are 0. Higher number = priority.\r\n   */\r\n  addHandler(\r\n    eventName: string,\r\n    handler: EventHandler,\r\n    userData: object | null = null,\r\n    priority: number = 0,\r\n  ) {\r\n    let events = this.events[eventName];\r\n    if (!events) {\r\n      this.events[eventName] = events = [];\r\n    }\r\n    if (handler && EventSource.isFunction(handler)) {\r\n      let index = events.length,\r\n        event = {\r\n          handler: handler,\r\n          userData: userData || null,\r\n          priority: priority || 0,\r\n        };\r\n      events[index] = event;\r\n      while (index > 0 && events[index - 1].priority < events[index].priority) {\r\n        events[index] = events[index - 1];\r\n        events[index - 1] = event;\r\n        index--;\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Remove a specific event handler for a given event.\r\n   * @function\r\n   * @param {String} eventName - Name of event for which the handler is to be removed.\r\n   * @param {EventHandler} handler - Function to be removed.\r\n   */\r\n  removeHandler(eventName: string, handler: EventHandler) {\r\n    const events = this.events[eventName],\r\n      handlers: EventHandler[] = [];\r\n    if (!events) {\r\n      return;\r\n    }\r\n    if (Array.isArray(events)) {\r\n      for (let i = 0; i < events.length; i++) {\r\n        if (events[i].handler !== handler) {\r\n          handlers.push(events[i]);\r\n        }\r\n      }\r\n      this.events[eventName] = handlers;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Get the amount of handlers registered for a given event.\r\n   * @param {String} eventName - Name of event to inspect.\r\n   * @returns {number} amount of events\r\n   */\r\n  numberOfHandlers(eventName: string) {\r\n    const events = this.events[eventName];\r\n    if (!events) {\r\n      return 0;\r\n    }\r\n    return events.length;\r\n  }\r\n\r\n  /**\r\n   * Remove all event handlers for a given event type. If no type is given all\r\n   * event handlers for every event type are removed.\r\n   * @function\r\n   * @param {String} eventName - Name of event for which all handlers are to be removed.\r\n   */\r\n  removeAllHandlers(eventName: string) {\r\n    if (eventName) {\r\n      this.events[eventName] = [];\r\n    } else {\r\n      for (const eventType in this.events) {\r\n        this.events[eventType] = [];\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Get a function which iterates the list of all handlers registered for a given event, calling the handler for each.\r\n   * @function\r\n   * @param {String} eventName - Name of event to get handlers for.\r\n   */\r\n  getHandler(eventName: string) {\r\n    let events = this.events[eventName];\r\n    if (!events || !events.length) {\r\n      return null;\r\n    }\r\n    events = events.length === 1 ? [events[0]] : Array.apply(null, events);\r\n    return function (source: any, args: any) {\r\n      let i,\r\n        length = events.length;\r\n      for (i = 0; i < length; i++) {\r\n        if (events[i]) {\r\n          args.eventSource = source;\r\n          args.userData = events[i].userData;\r\n          events[i].handler(args);\r\n        }\r\n      }\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Get a function which iterates the list of all handlers registered for a given event,\r\n   * calling the handler for each and awaiting async ones.\r\n   * @function\r\n   * @param {String} eventName - Name of event to get handlers for.\r\n   */\r\n  getAwaitingHandler(eventName) {\r\n    let events = this.events[eventName];\r\n    if (!events || !events.length) {\r\n      return null;\r\n    }\r\n    events = events.length === 1 ? [events[0]] : Array.apply(null, events);\r\n\r\n    return function (source, args) {\r\n      // We return a promise that gets resolved after all the events finish.\r\n      // Returning loop result is not correct, loop promises chain dynamically\r\n      // and outer code could process finishing logics in the middle of event loop.\r\n      return new Promise((resolve) => {\r\n        const length = events.length;\r\n        function loop(index) {\r\n          if (index >= length || !events[index]) {\r\n            resolve('Resolved!');\r\n            return null;\r\n          }\r\n          args.eventSource = source;\r\n          args.userData = events[index].userData;\r\n          let result = events[index].handler(args);\r\n          result =\r\n            !result || EventSource.type(result) !== 'promise'\r\n              ? Promise.resolve()\r\n              : result;\r\n          return result.then(() => loop(index + 1));\r\n        }\r\n        loop(0);\r\n      });\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Trigger an event, optionally passing additional information.\r\n   * @function\r\n   * @param {String} eventName - Name of event to register.\r\n   * @param {Object} eventArgs - Event-specific data.\r\n   */\r\n  raiseEvent(eventName: string, eventArgs?: any) {\r\n    const handler = this.getHandler(eventName);\r\n    if (handler) {\r\n      return handler(this, eventArgs || {});\r\n    }\r\n    return undefined;\r\n  }\r\n\r\n  /**\r\n   * Trigger an event, optionally passing additional information.\r\n   * This events awaits every asynchronous or promise-returning function.\r\n   * @param {String} eventName - Name of event to register.\r\n   * @param {Object} eventArgs - Event-specific data.\r\n   * @return {Promise|undefined} - Promise resolved upon the event completion.\r\n   */\r\n  raiseEventAwaiting(eventName, eventArgs) {\r\n    //uncomment if you want to get a log of all events\r\n    //$.console.log( \"Awaiting event fired:\", eventName );\r\n\r\n    const awaitingHandler = this.getAwaitingHandler(eventName);\r\n    if (awaitingHandler) {\r\n      return awaitingHandler(this, eventArgs || {});\r\n    }\r\n    return Promise.resolve('No handler for this event registered.');\r\n  }\r\n\r\n  static class2type = {\r\n    '[object Boolean]': 'boolean',\r\n    '[object Number]': 'number',\r\n    '[object String]': 'string',\r\n    '[object Function]': 'function',\r\n    '[object AsyncFunction]': 'function',\r\n    '[object Promise]': 'promise',\r\n    '[object Array]': 'array',\r\n    '[object Date]': 'date',\r\n    '[object RegExp]': 'regexp',\r\n    '[object Object]': 'object',\r\n  };\r\n\r\n  /**\r\n   * Taken from jQuery 1.6.1\r\n   * @function isFunction\r\n   * @memberof OpenSeadragon\r\n   * @see {@link http://www.jquery.com/ jQuery}\r\n   */\r\n  static isFunction(obj: object) {\r\n    return this.type(obj) === 'function';\r\n  }\r\n\r\n  /**\r\n   * Taken from jQuery 1.6.1\r\n   * @function type\r\n   * @memberof OpenSeadragon\r\n   * @see {@link http://www.jquery.com/ jQuery}\r\n   */\r\n  static type(obj: object) {\r\n    return obj === null || obj === undefined\r\n      ? String(obj)\r\n      : this.class2type[obj.toString() as keyof typeof this.class2type] ||\r\n          (typeof obj === 'function' ? 'function' : 'object');\r\n  }\r\n}\r\n","// Source: https://github.com/prettymuchbryce/http-status-codes/blob/master/src/status-codes.ts\r\nexport const STATUS_CODES = {\r\n  100: 'Continue',\r\n  101: 'Switching protocols',\r\n  102: 'Processing',\r\n  103: 'Early Hints',\r\n  200: 'Ok',\r\n  201: 'Created',\r\n  202: 'Accepted',\r\n  203: 'Non Authoritative Information',\r\n  204: 'No Content',\r\n  205: 'Reset Content',\r\n  206: 'Partial Content',\r\n  207: 'Multi Status',\r\n  300: 'Multiple Choices',\r\n  301: 'Moved Permanently',\r\n  302: 'Moved Temporarily',\r\n  303: 'See Other',\r\n  304: 'Not Modified',\r\n  305: 'Use Proxy',\r\n  307: 'Temporary Redirect',\r\n  308: 'Permanent Redirect',\r\n  400: 'Bad Request',\r\n  401: 'Unauthorized',\r\n  402: 'Payment Required',\r\n  403: 'Forbidden',\r\n  404: 'Not Found',\r\n  405: 'Method Not Allowed',\r\n  406: 'Not Acceptable',\r\n  407: 'Proxy Authentication Required',\r\n  408: 'Request Timeout',\r\n  409: 'Conflict',\r\n  410: 'Gone',\r\n  411: 'Length Required',\r\n  412: 'Precondition Failed',\r\n  413: 'Request Too Long',\r\n  414: 'Request Uri Too Long',\r\n  415: 'Unsupported Media Type',\r\n  416: 'Requested Range Not Satisfiable',\r\n  417: 'Expectation Failed',\r\n  /**\r\n   * Official Documentation @ https://tools.ietf.org/html/rfc2324#section-2.3.2\r\n   *\r\n   * Any attempt to brew coffee with a teapot should result in the error code \"418 I'm a teapot\". The resulting entity body May be short and stout.\r\n   */\r\n  418: 'Im A Teapot',\r\n  419: 'Insufficient Space On Resource',\r\n  420: 'Method Failure',\r\n  421: 'Misdirected Request',\r\n  422: 'Unprocessable Entity',\r\n  423: 'Locked',\r\n  424: 'Failed Dependency',\r\n  426: 'Upgrade Required',\r\n  428: 'Precondition Required',\r\n  429: 'Too Many Requests',\r\n  431: 'Request Header Fields Too Large',\r\n  451: 'Unavailable For Legal Reasons',\r\n  500: 'Internal Server Error',\r\n  501: 'Not Implemented',\r\n  502: 'Bad Gateway',\r\n  503: 'Service Unavailable',\r\n  504: 'Gateway Timeout',\r\n  505: 'Http Version Not Supported',\r\n  507: 'Insufficient Storage',\r\n  511: 'Network Authentication Required',\r\n};\r\n","import { EventSource } from './events';\r\nimport { STATUS_CODES } from './status-codes';\r\nimport {DefaultIntegrationOptions} from \"./v3/integration/default\";\r\n\r\nexport interface EmpationAPIOptions {\r\n  anonymousUserId?: string;\r\n  workbenchApiUrl: string;\r\n  apiRootPath?: string;\r\n  integrationOptions?: DefaultIntegrationOptions\r\n}\r\n\r\ntype ResponseType = 'json' | 'blob' | 'text';\r\n\r\nexport interface RawOptions {\r\n  body?: object | string;\r\n  query?: any;\r\n  method?: string;\r\n  headers?: { [key: string]: string };\r\n  responseType?: ResponseType;\r\n}\r\n\r\n//https://gist.github.com/TooTallNate/4fd641f820e1325695487dfd883e5285\r\nfunction httpErrorToName(code: number): string {\r\n  const suffix =\r\n    ((code / 100) | 0) === 4 || ((code / 100) | 0) === 5 ? 'error' : '';\r\n  let name = ` ${String(STATUS_CODES[code as keyof typeof STATUS_CODES] || `HTTP Code ${code}`).replace(/error$/i, '')} ${suffix}`;\r\n  return name\r\n    .split(' ')\r\n    .reduce(\r\n      (acc, c) => acc + (c ? c.charAt(0).toUpperCase() + c.slice(1) : ''),\r\n    );\r\n}\r\n\r\nexport class HTTPError extends Error {\r\n  statusCode: number;\r\n  [key: string]: any;\r\n\r\n  public constructor(\r\n    code: number,\r\n    message: string,\r\n    extras?: Record<string, any>,\r\n  ) {\r\n    super(\r\n      message ||\r\n        STATUS_CODES[code as keyof typeof STATUS_CODES] ||\r\n        `HTTP Code ${code}`,\r\n    );\r\n    if (arguments.length >= 3 && extras) {\r\n      // noinspection TypeScriptValidateTypes\r\n      Object.assign(this, extras);\r\n    }\r\n    this.name = httpErrorToName(code);\r\n    this.statusCode = code;\r\n  }\r\n}\r\n\r\nexport interface RawApiOptions {}\r\n\r\nexport class RawAPI {\r\n  public url: string;\r\n  public options: RawOptions;\r\n  constructor(url: string, options: RawApiOptions = {}) {\r\n    this.url = url;\r\n    this.options = options;\r\n  }\r\n\r\n  private _parseQueryParams(params: string | { [key: string]: string }) {\r\n    if (params) {\r\n      if (typeof params === 'string') return params;\r\n\r\n      //cleanup plain objects\r\n      if (params.constructor === Object || params.constructor === undefined) {\r\n        for (let k in params) {\r\n          const v = params[k];\r\n          if (v === null || v === undefined) delete params[k];\r\n        }\r\n      }\r\n      return `?${new URLSearchParams(params)}`;\r\n    }\r\n    return '';\r\n  }\r\n\r\n  private async _fetch(url: string, options: RawOptions): Promise<any> {\r\n    const response = await fetch(url, {\r\n      method: options.method,\r\n      headers: options.headers,\r\n      body: options.body,\r\n    } as RequestInit);\r\n\r\n    let result;\r\n    try {\r\n      result = await response[options.responseType || 'json']();\r\n    } catch (e) {\r\n      throw new HTTPError(\r\n        500,\r\n        `Failed to parse response data. Original status: ${response.status} | ${response.statusText}`,\r\n        {\r\n          url: url,\r\n          error: e,\r\n        },\r\n      );\r\n    }\r\n\r\n    if (!response.ok) {\r\n      throw new HTTPError(response.status, response.statusText, {\r\n        payload: result\r\n      });\r\n    }\r\n    return result;\r\n  }\r\n\r\n  async http(endpoint: string, options: RawOptions): Promise<any> {\r\n    const hasBody = !!options.body;\r\n    options.method = options.method || (hasBody ? 'POST' : 'GET');\r\n    if (!endpoint.startsWith('/')) {\r\n      endpoint = `/${endpoint}`;\r\n    }\r\n    options.query = this._parseQueryParams(options.query);\r\n    options.headers = options.headers || {};\r\n    options.headers['Content-Type'] = 'application/json';\r\n    if (options.body && typeof options.body !== 'string') {\r\n      options.body = JSON.stringify(options.body);\r\n    } else options.body = undefined;\r\n\r\n    return await this._fetch(this.url + endpoint + options.query, options);\r\n  }\r\n}\r\n\r\n//todo consider implementing 'observe' method that registers and watches certain property list..\r\nexport class AbstractAPI extends EventSource {\r\n  //todo try figuring out how to print the class name too\r\n  //https://stackoverflow.com/questions/280389/how-do-you-find-out-the-caller-function-in-javascript\r\n  private getCallerName() {\r\n    // Get stack array\r\n    const orig = Error.prepareStackTrace;\r\n    Error.prepareStackTrace = (error, stack) => stack;\r\n    const { stack } = new Error();\r\n    Error.prepareStackTrace = orig;\r\n\r\n    const caller = stack?.[2];\r\n    return caller ? caller : 'unknown context';\r\n  }\r\n\r\n  requires(name: string, value: any): void {\r\n    if (!value) {\r\n      throw `ArgumentError[${this.getCallerName()}] ${name} is missing - required property!`;\r\n    }\r\n  }\r\n}\r\n","export interface JwtTokenBase {\r\n  sub: string;\r\n  exp: number;\r\n}\r\n\r\nexport interface JwtToken extends JwtTokenBase {\r\n  iat: number;\r\n  jti: string;\r\n  iss: string;\r\n  aud: string;\r\n  typ: string;\r\n  azp: string;\r\n}\r\n\r\nexport function getJwtTokenExpiresTimeout(token: JwtTokenBase) {\r\n  //timeout OR 300 secs\r\n  return token.exp * 1e3 - Date.now() || 300e3;\r\n}\r\n\r\nexport interface ScopeToken extends JwtTokenBase {\r\n  token_id: number;\r\n}\r\n\r\nexport function parseJwtToken(token: string): JwtTokenBase {\r\n  return JSON.parse(atob(token.split('.')[1]));\r\n}\r\n\r\nexport function sleep(ms: number): Promise<void> {\r\n  return new Promise((resolve) => setTimeout(resolve, ms));\r\n}\r\n\r\nexport class Logger {\r\n  static error(...args: any): void {\r\n    console.error('E:EmpationAPI', ...args);\r\n  }\r\n  static warn(...args: any): void {\r\n    console.warn('W:EmpationAPI', ...args);\r\n  }\r\n  static info(...args: any): void {\r\n    console.info('I:EmpationAPI', ...args);\r\n  }\r\n  static debug(...args: any): void {\r\n    console.debug('D:EmpationAPI', ...args);\r\n  }\r\n}\r\n","import { AbstractAPI, EmpationAPIOptions, RawAPI, RawOptions } from './base';\r\nimport { ScopeAPI } from './scope';\r\nimport {\r\n  getJwtTokenExpiresTimeout,\r\n  JwtTokenBase,\r\n  parseJwtToken,\r\n} from './utils';\r\n\r\nexport abstract class RootContext {\r\n  protected abstract context: RootAPI;\r\n  protected abstract data: any;\r\n}\r\n\r\nexport interface RootAPIOptions {\r\n  anonymousUserId: string;\r\n  workbenchApiUrl: string;\r\n  apiUrl: string;\r\n  apiRootPath: string;\r\n}\r\n\r\n// BaseAPI implements AbstractAPI over /v[version]\r\nexport abstract class RootAPI extends AbstractAPI {\r\n  // RawAPI implements access to the http endpoints\r\n  protected abstract raw: RawAPI;\r\n  // default ScopeAPI implements AbstractAPI over /v[version]/scopes\r\n  protected abstract defaultScopeKey: string;\r\n\r\n  // Map of ScopeAPI, that implement AbstractAPI over /v[version]/scopes, allows keeping multiple scopes open at once\r\n  abstract scopes: Map<string, ScopeAPI>;\r\n\r\n  // Properties\r\n  abstract version: string;\r\n  abstract rootURI: string;\r\n  options: RootAPIOptions;\r\n  cached: object;\r\n  accessToken: JwtTokenBase | null = null;\r\n\r\n  private _userId: string;\r\n  private _tokenExpires: number = 0;\r\n  private _rawToken: string = '';\r\n\r\n  protected constructor(options: EmpationAPIOptions) {\r\n    super();\r\n\r\n    if (!options.workbenchApiUrl) {\r\n      throw 'WB Api url is required!';\r\n    }\r\n\r\n    let apiUrl;\r\n    if (!options.apiRootPath) {\r\n      apiUrl = options.workbenchApiUrl;\r\n    } else if (!options.apiRootPath.startsWith('/')) {\r\n      apiUrl = `${options.workbenchApiUrl}/${options.apiRootPath}`;\r\n    } else {\r\n      apiUrl = `${options.workbenchApiUrl}${options.apiRootPath}`;\r\n    }\r\n    if (apiUrl.endsWith('/')) {\r\n      apiUrl = apiUrl.slice(0, -1);\r\n    }\r\n    this.options = {\r\n      apiUrl,\r\n      workbenchApiUrl: options.workbenchApiUrl,\r\n      anonymousUserId: options.anonymousUserId || 'anonymous',\r\n      apiRootPath: options.apiRootPath || '',\r\n    };\r\n    this._userId = this.options.anonymousUserId;\r\n    this.cached = {};\r\n  }\r\n\r\n  /**\r\n   * Change the User actor for the API. Note: the api will\r\n   * reset it's whole state.\r\n   * @param token setup context from object\r\n   * @param withEvent\r\n   */\r\n  from(token: string, withEvent = true): void {\r\n    if (!token) {\r\n      return this.reset();\r\n    }\r\n    this._rawToken = token;\r\n    withEvent = withEvent && !this.accessToken; //fire event only when we configure new session\r\n    this.accessToken = parseJwtToken(token) as JwtTokenBase;\r\n    const tokenTimeout = getJwtTokenExpiresTimeout(this.accessToken);\r\n    this._tokenExpires = Date.now() + tokenTimeout / 2;\r\n    let userId = this.accessToken.sub;\r\n    if (!userId)\r\n      throw 'Invalid User ID! Must be valid string shorter than 50 characters!';\r\n    if (userId.length > 50) {\r\n      console.warn(\r\n        'User ID exceeded 50 characters! Using User ID shortened to first 50 characters!',\r\n      );\r\n      userId = userId.slice(0, 50);\r\n    }\r\n    if (this.userId === userId) return;\r\n    this._userId = userId;\r\n    if (withEvent) this.raiseEvent('init');\r\n  }\r\n\r\n  /**\r\n   * Change the User actor for the API, without providing token. Used in no-token configuration. Note: the api will\r\n   * reset it's whole state.\r\n   * @param token setup context from object\r\n   * @param withEvent\r\n   */\r\n  use(userId: string, withEvent = true): void {\r\n    withEvent = withEvent && !this._userId; //fire event only when we configure new session\r\n    this.reset();\r\n    if (!userId || userId.length > 50)\r\n      throw 'Invalid User ID! Must be valid string shorter than 50 characters!';\r\n    this._userId = userId;\r\n    if (withEvent) this.raiseEvent('init_no_token');\r\n  }\r\n\r\n  reset(): void {\r\n    this._rawToken = '';\r\n    this._tokenExpires = 0;\r\n    this.accessToken = null;\r\n    this._userId = this.options.anonymousUserId;\r\n    this.defaultScopeKey = '';\r\n    this.scopes.forEach((scp) => scp.reset());\r\n    this.scopes.clear();\r\n    this.raiseEvent('reset');\r\n  }\r\n\r\n  get userId(): string {\r\n    return this._userId;\r\n  }\r\n\r\n  get rawToken(): string {\r\n    return this._rawToken;\r\n  }\r\n\r\n  async rawQuery(endpoint: string, options?: RawOptions): Promise<any> {\r\n    if (!this._userId) {\r\n      throw \"User must be configured to access Empaia API: either provide a valid 'anonymous' user ID through env variables, or configure the Root API with a valid token.\";\r\n    }\r\n\r\n    if (this._tokenExpires > 0 && Date.now() > this._tokenExpires) {\r\n      const eventObject = { newToken: '' };\r\n      /**\r\n       * @event token-refresh\r\n       * Awaiting event. Provide newToken value in the event handler argument params object\r\n       * for the root API to consume.\r\n       */\r\n      await this.raiseEventAwaiting('token-refresh', eventObject);\r\n      this.from(eventObject.newToken);\r\n    }\r\n  }\r\n}\r\n","import { AbstractAPI, RawAPI, RawOptions } from './base';\r\n\r\nexport abstract class ScopeContext {\r\n  protected abstract context: ScopeAPI;\r\n  protected abstract data: any;\r\n}\r\n\r\n/**\r\n * Scope Binds Examination and User.\r\n */\r\nexport abstract class ScopeAPI extends AbstractAPI {\r\n  protected abstract raw: RawAPI;\r\n\r\n  /**\r\n   * Change the active Scope for the API. Note: the scope state will reset.\r\n   * @param caseId case id - manages the active scope\r\n   * @param appId manages the active scope through examination<user, app>\r\n   *          if undefined, the examination is managed internally\r\n   */\r\n  abstract use(caseId: string, appId?: string): Promise<void>;\r\n\r\n  /**\r\n   * Change the active Scope for the API. Note: the scope state will reset.\r\n   * @param examination setup context from object\r\n   */\r\n  abstract from(examination: object): Promise<void>;\r\n\r\n  abstract reset(): void;\r\n\r\n  abstract rawQuery(endpoint: string, options?: RawOptions): Promise<any>;\r\n}\r\n","import { RootAPI, RootContext } from '../../root';\r\nimport Root from './root';\r\nimport { AppList } from './types/app-list';\r\nimport { AppQuery } from './types/app-query';\r\nimport { AppOutput } from './types/app-output';\r\n\r\nexport default class Apps extends RootContext {\r\n  protected context: RootAPI;\r\n  protected data: AppList | null = null;\r\n  private _defaultApp: AppOutput | null = null;\r\n\r\n  constructor(context: Root) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  async list(): Promise<AppList> {\r\n    return (this.data = (await this.context.rawQuery('/apps/query', {\r\n      method: 'PUT',\r\n      body: {\r\n        apps: null,\r\n        tissues: null,\r\n        stains: null,\r\n        job_modes: null,\r\n      },\r\n    })) as AppList);\r\n  }\r\n\r\n  async query(query: AppQuery): Promise<AppList> {\r\n    return (this.data = (await this.context.rawQuery('/apps/query', {\r\n      method: 'PUT',\r\n      body: query,\r\n    })) as AppList);\r\n  }\r\n\r\n  async default(): Promise<AppOutput> {\r\n    if (!this.data) await this.list();\r\n    for (let app of this.data!.items) {\r\n      if (app.name_short === 'MAP3' && app.vendor_name === 'rationai') {\r\n        this._defaultApp = app;\r\n        break;\r\n      }\r\n    }\r\n    if (!this._defaultApp)\r\n      throw 'Default APP not present in the infrastructure! Was it imported?';\r\n    return this._defaultApp;\r\n  }\r\n}\r\n","/**\r\n * Use for converting strings representing number to a number.\r\n * @param variable Number or string representing a number.\r\n */\r\nconst getNumber = (variable: string | number) => {\r\n  if (typeof variable === 'string') {\r\n    variable = Number(variable);\r\n  }\r\n\r\n  return variable;\r\n};\r\n\r\nexport const getYearFromEpochTime = (epochTime: number | string) => {\r\n  return new Date(getNumber(epochTime) * 1000).getFullYear();\r\n};\r\n\r\nexport const getMonthFromEpochTime = (epochTime: number | string) => {\r\n  return new Date(getNumber(epochTime) * 1000).getMonth();\r\n};\r\n\r\nexport const getDayFromEpochTime = (epochTime: number | string) => {\r\n  return new Date(getNumber(epochTime) * 1000).getDate();\r\n};\r\n\r\n/**\r\n * Match string on specific substring and value.\r\n * @param str String value.\r\n * @param separator Regex that should match the provided value.\r\n * @param groupIdx Index of a group matched by the regex.\r\n * @param value Value the matched group should have.\r\n */\r\nexport const matchStringOnSeparatorGroup = (\r\n  str: string,\r\n  separator: string,\r\n  groupIdx: number,\r\n  value: string,\r\n) => {\r\n  const matches = new RegExp(separator).exec(str);\r\n  if (!matches || groupIdx < 1 || groupIdx >= matches.length) return false;\r\n  return matches[groupIdx] === value;\r\n};\r\n\r\n/**\r\n * Match string if it contains some of the specified tokens.\r\n * @param stringToMatch String value.\r\n * @param tokenString String containing tokens/words split by a \" \".\r\n */\r\nexport const matchStringOnTokens = (\r\n  stringToMatch: string,\r\n  tokenString: string,\r\n) => {\r\n  const tokens = tokenString\r\n    .split(' ')\r\n    .filter(Boolean)\r\n    .map((token) => `(?=.*\\\\b${token}\\\\b)`);\r\n\r\n  const searchTermRegex = new RegExp(tokens.join(''), 'gim');\r\n  return stringToMatch.match(searchTermRegex) !== null;\r\n};\r\n\r\n/**\r\n * Group by for javascript array.\r\n * @param arr Array of objects.\r\n * @param key Function to get value from object by which the objects are grouped.\r\n */\r\nexport const groupBy = <T, K extends keyof any>(arr: T[], key: (i: T) => K) =>\r\n  arr.reduce(\r\n    (groups, item) => {\r\n      (groups[key(item)] ||= []).push(item);\r\n      return groups;\r\n    },\r\n    {} as Record<K, T[]>,\r\n  );\r\n\r\n/**\r\n * Constructor Type\r\n */\r\nexport type Constructor<T> = new (...args: any[]) => T;\r\n\r\n/**\r\n * Load default class by a string\r\n * @param classPath\r\n * @param name\r\n */\r\nexport async function loadClass<T>(classPath: string, name: string = 'default'): Promise<Constructor<T>> {\r\n  const module = await import(classPath);\r\n  const ClassConstructor = module[name];\r\n  if (ClassConstructor === undefined) {\r\n    throw new Error(`ClassPath ${classPath} does not export ${name}!`);\r\n  }\r\n  return ClassConstructor as Constructor<T>;\r\n}\r\n\r\nexport async function instantiate<T>(props: any, classPath: string, name: string = 'default'): Promise<T> {\r\n  const Cls = await loadClass<T>(classPath, name);\r\n  return new Cls(props);\r\n}\r\n","import Cases from '../root/cases';\r\nimport { Case } from '../root/types/case';\r\nimport { CaseH } from './types/case-h';\r\nimport { CaseHierarchy } from './types/case-hierarchy-result';\r\nimport { CaseSearchParams } from './types/case-search-params';\r\nimport {\r\n  getDayFromEpochTime,\r\n  getMonthFromEpochTime,\r\n  getYearFromEpochTime,\r\n  groupBy,\r\n  matchStringOnTokens,\r\n} from './utils';\r\nimport {AuthIntegration} from \"../integration\";\r\n\r\nexport type CaseTissuesStains = {\r\n  name: string;\r\n  locName: string;\r\n};\r\n\r\nexport type HierarchyNameOverrides = {\r\n  [key: string]: {\r\n    [key: string]: string;\r\n  };\r\n};\r\n\r\nexport default class CaseExplorer {\r\n  protected context: Cases;\r\n  protected customCases: CaseH[] | null = null;\r\n  protected caseHierarchy: CaseHierarchy | null = null;\r\n  protected caseTissues: CaseTissuesStains[] | null = null;\r\n  protected caseStains: CaseTissuesStains[] | null = null;\r\n  protected integration: AuthIntegration;\r\n\r\n  identifierSeparator: string = '';\r\n  hierarchySpec: string[] = [];\r\n  hierarchyNameOverrides: HierarchyNameOverrides = {};\r\n\r\n  constructor(context: Cases, integration: AuthIntegration) {\r\n    this.context = context;\r\n    this.integration = integration;\r\n  }\r\n\r\n  /**\r\n   * Configure CaseExplorer.\r\n   * @param identifierSeparator Regex matching local_id and its parts.\r\n   * @param hierarchySpec Array of hierarchy keys defining hierarchy.\r\n   * @param hierarchyNameOverrides Override specific values of specific keys to use as names in hierarchy.\r\n   */\r\n  use(\r\n    identifierSeparator: string,\r\n    hierarchySpec: string[],\r\n    hierarchyNameOverrides: HierarchyNameOverrides = {},\r\n  ): void {\r\n    this.hierarchySpec = hierarchySpec;\r\n    this.identifierSeparator = identifierSeparator;\r\n    this.hierarchyNameOverrides = hierarchyNameOverrides;\r\n  }\r\n\r\n  /**\r\n   * Returns cases extended with path in the specified hierarchy.\r\n   */\r\n  private async getCustomCases() {\r\n    if (!this.customCases) {\r\n      this.customCases = (await this.context.list()).items.map((caseObj) => {\r\n        return {\r\n          ...caseObj,\r\n          pathInHierarchy: this.getCaseHierarchyPath(caseObj),\r\n        };\r\n      });\r\n    }\r\n    return this.customCases;\r\n  }\r\n\r\n  /**\r\n   * Returns case's path in the specified hierarchy.\r\n   * @param caseObj EMPAIA Case.\r\n   */\r\n  getCaseHierarchyPath(caseObj: Case) {\r\n    if (!this.identifierSeparator || !this.hierarchySpec) {\r\n      throw `ArgumentError[CaseExplorer] identifierSeparator or hierarchySpec is missing - required property!`;\r\n    }\r\n    let pathFinished = false;\r\n    return this.hierarchySpec.reduce((prev, curr) => {\r\n      const val = this.getCaseValue(curr, caseObj);\r\n\r\n      const returnVal = pathFinished ? prev : `${prev}/${val}`;\r\n      if (val === 'OTHER') {\r\n        pathFinished = true;\r\n      }\r\n      return returnVal;\r\n    }, '');\r\n  }\r\n\r\n  // private async getCaseHierarchyPath(caseObj: Case): Promise<{pathInHierarchy: string, hierarchy: CaseHElement}> {\r\n  //   if (!this.identifierSeparator || !this.hierarchySpec) {\r\n  //     throw `ArgumentError[CaseExplorer] identifierSeparator or hierarchySpec is missing - required property!`;\r\n  //   }\r\n  //\r\n  //   let currentNode = undefined;\r\n  //   const result = {\r\n  //     pathInHierarchy: \"\",\r\n  //     hierarchy: currentNode\r\n  //   };\r\n  //   for (let specKey of this.hierarchySpec) {\r\n  //     const val = this.getCaseValue(specKey, caseObj);\r\n  //     const translated = this.integration.translatePathSpec(specKey,\r\n  //         (typeof val === \"string\") ? val : val.toString());\r\n  //\r\n  //     result.pathInHierarchy = `${result.pathInHierarchy}/${val}`;\r\n  //     currentNode = {\r\n  //       id: val,\r\n  //       name: translated,\r\n  //       currentNode: currentNode\r\n  //     };\r\n  //\r\n  //     if (val === 'OTHER') {\r\n  //       break;\r\n  //     }\r\n  //   }\r\n  //   return result;\r\n  // }\r\n\r\n  /**\r\n   * Returns single case.\r\n   * @param caseId ID of a case.\r\n   */\r\n  async getCase(caseId: string): Promise<CaseH> {\r\n    let caseObj: Case | undefined;\r\n    if (this.customCases) {\r\n      caseObj = this.customCases.find((cs) => cs.id === caseId);\r\n    }\r\n\r\n    if (!caseObj) {\r\n      caseObj = await this.context.get(caseId);\r\n    }\r\n\r\n    return { ...caseObj, pathInHierarchy: this.getCaseHierarchyPath(caseObj) };\r\n  }\r\n\r\n  /**\r\n   * Returns case's value, can be a simple attribute, or some derived value.\r\n   * @param key Key specifying value that can be extracted from a case.\r\n   * @param cs Case object.\r\n   */\r\n  private getCaseValue(key: string, cs: Case) {\r\n    switch (key) {\r\n      case 'year': {\r\n        return this.getCaseYear(cs);\r\n      }\r\n      case 'month': {\r\n        return this.getCaseMonth(cs);\r\n      }\r\n      case 'day': {\r\n        return this.getCaseDay(cs);\r\n      }\r\n      case 'description': {\r\n        return this.getCaseDescription(cs);\r\n      }\r\n      case 'tissues': {\r\n        return this.getCaseTissues(cs);\r\n      }\r\n      case 'stains': {\r\n        return this.getCaseStains(cs);\r\n      }\r\n      default: {\r\n        if (key.slice(0, 8) === 'id_part_' && !isNaN(Number(key.slice(8)))) {\r\n          return this.getCaseIdentifierPart(cs, Number(key.slice(8)));\r\n        }\r\n        throw `KeyError[CaseExplorer] \\\"${key}\\\" is not supported!`;\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Evaluates case's value against a provided value.\r\n   * @param key Key specifying value that can be extracted from a case.\r\n   * @param evalValue Value for comparison/evaluation.\r\n   * @param cs Case object.\r\n   */\r\n  private evaluateCaseValue(\r\n    key: string,\r\n    evalValue: string | string[],\r\n    cs: Case,\r\n  ) {\r\n    const caseValue = this.getCaseValue(key, cs);\r\n    switch (key) {\r\n      case 'year': {\r\n        return this.evaulateCaseYear(caseValue as string, evalValue as string);\r\n      }\r\n      case 'month': {\r\n        return this.evaulateCaseMonth(caseValue as string, evalValue as string);\r\n      }\r\n      case 'day': {\r\n        return this.evaulateCaseDay(caseValue as string, evalValue as string);\r\n      }\r\n      case 'description': {\r\n        return this.evaluateCaseDescription(\r\n          caseValue as string,\r\n          evalValue as string,\r\n        );\r\n      }\r\n      case 'tissues': {\r\n        return this.evaluateCaseTissues(\r\n          caseValue as string[],\r\n          evalValue as string | string[],\r\n        );\r\n      }\r\n      case 'stains': {\r\n        return this.evaluateCaseStains(\r\n          caseValue as string[],\r\n          evalValue as string | string[],\r\n        );\r\n      }\r\n      default: {\r\n        // any invalid key will be caught in getCaseValue call\r\n        return this.evaulateCaseIdentifierPart(\r\n          caseValue as string,\r\n          evalValue as string,\r\n        );\r\n      }\r\n    }\r\n  }\r\n\r\n  private getCaseYear(cs: Case): string {\r\n    return getYearFromEpochTime(cs.created_at).toString();\r\n  }\r\n  private getCaseMonth(cs: Case): string {\r\n    return getMonthFromEpochTime(cs.created_at).toString();\r\n  }\r\n  private getCaseDay(cs: Case): string {\r\n    return getDayFromEpochTime(cs.created_at).toString();\r\n  }\r\n  private getCaseIdentifierPart(cs: Case, partIdx: number): string {\r\n    if (!this.identifierSeparator) {\r\n      throw `ArgumentError[CaseExplorer] identifierSeparator is missing - required property!`;\r\n    }\r\n    const parts = new RegExp(this.identifierSeparator).exec(cs.local_id || '');\r\n    if (!parts) return 'OTHER';\r\n    if (partIdx < 1 || partIdx >= parts.length)\r\n      throw `KeyError[CaseExplorer] invalid key \\\"id_part_<index>\\\", group index is not valid!`;\r\n    return parts[partIdx];\r\n  }\r\n  private getCaseDescription(cs: Case): string {\r\n    return cs.description || '';\r\n  }\r\n  private getCaseTissues(cs: Case): string[] {\r\n    return Object.keys(cs.tissues);\r\n  }\r\n  private getCaseStains(cs: Case): string[] {\r\n    return Object.keys(cs.stains);\r\n  }\r\n\r\n  private evaulateCaseYear(value: string, evalValue: string): boolean {\r\n    return value === evalValue;\r\n  }\r\n  private evaulateCaseMonth(value: string, evalValue: string): boolean {\r\n    return value === evalValue;\r\n  }\r\n  private evaulateCaseDay(value: string, evalValue: string): boolean {\r\n    return value === evalValue;\r\n  }\r\n  private evaulateCaseIdentifierPart(\r\n    value: string,\r\n    evalValue: string,\r\n  ): boolean {\r\n    return value === evalValue;\r\n  }\r\n  private evaluateCaseDescription(value: string, evalValue: string): boolean {\r\n    return matchStringOnTokens(value, evalValue);\r\n  }\r\n  private evaluateCaseTissues(\r\n    value: string[],\r\n    evalValue: string | string[],\r\n  ): boolean {\r\n    if (!(evalValue instanceof Array)) {\r\n      evalValue = [evalValue];\r\n    }\r\n    // ALL searched tissues are present in case\r\n    return evalValue.every((tissue) => value.includes(tissue));\r\n\r\n    // SOME searched tissues are present in case\r\n    // return evalValue.some((tissue) => value.includes(tissue))\r\n  }\r\n  private evaluateCaseStains(\r\n    value: string[],\r\n    evalValue: string | string[],\r\n  ): boolean {\r\n    if (!(evalValue instanceof Array)) {\r\n      evalValue = [evalValue];\r\n    }\r\n    // ALL searched stains are present in case\r\n    return evalValue.every((stain) => value.includes(stain));\r\n\r\n    // SOME searched stains are present in case\r\n    // return evalValue.some((stain) => value.includes(stain))\r\n  }\r\n\r\n  /**\r\n   * Recursively constructs a hierarchy by single levels\r\n   */\r\n  private async hierarchyLevel(\r\n    keys: string[],\r\n    keyIdx: number,\r\n    cases: Case[],\r\n    currentHierarchyPath: string,\r\n    id?: string,\r\n    name?: string,\r\n    parent?: CaseHierarchy,\r\n  ): Promise<CaseHierarchy> {\r\n    if (keyIdx >= keys.length) {\r\n      return {\r\n        levelId: id,\r\n        levelName: name,\r\n        lastLevel: true,\r\n        parent: parent,\r\n        items: cases.map((caseObj) => {\r\n          return { ...caseObj, pathInHierarchy: currentHierarchyPath };\r\n        }),\r\n      };\r\n    }\r\n    // grouping by array values(tissues, stains) is not expected, but works by grouping on first value of array\r\n    const groups = groupBy(cases, (cs) => {\r\n      const value = this.getCaseValue(keys[keyIdx], cs);\r\n      if (Array.isArray(value)) {\r\n        return value[0] || '';\r\n      }\r\n      return value;\r\n    });\r\n\r\n    const result: CaseHierarchy = {\r\n      levelName: name,\r\n      levelId: id,\r\n      lastLevel: false,\r\n      items: [],\r\n    };\r\n    result.items = await Promise.all(Object.keys(groups).map(async (itemId): Promise<CaseHierarchy> => {\r\n      let overrideName =\r\n          this.hierarchyNameOverrides[keys[keyIdx]]?.[itemId] ||\r\n          await this.integration.translatePathSpec(keys[keyIdx], itemId) ||\r\n          itemId;\r\n\r\n      if (itemId === 'OTHER') {\r\n        const child = await this.hierarchyLevel(\r\n            keys,\r\n            keys.length,\r\n            groups[itemId],\r\n            `${currentHierarchyPath}/${overrideName}`,\r\n            itemId,\r\n            overrideName,\r\n        );\r\n        child.parent = result;\r\n        return child;\r\n      }\r\n      const child = await this.hierarchyLevel(\r\n          keys,\r\n          keyIdx + 1,\r\n          groups[itemId],\r\n          `${currentHierarchyPath}/${overrideName}`,\r\n          itemId,\r\n          overrideName,\r\n      );\r\n      child.parent = result;\r\n      return child;\r\n    }));\r\n    return result;\r\n  }\r\n\r\n  /**\r\n   * Constructs a hierarchy based on spec configured in the CaseExplorer class.\r\n   */\r\n  async hierarchy(): Promise<CaseHierarchy> {\r\n    if (!this.caseHierarchy) {\r\n      const cases = await this.getCustomCases();\r\n      this.caseHierarchy = await this.hierarchyLevel(\r\n        this.hierarchySpec,\r\n        0,\r\n        cases,\r\n        '',\r\n      );\r\n    }\r\n    return this.caseHierarchy;\r\n  }\r\n\r\n  /**\r\n   * Search cases.\r\n   * @param query Search query.\r\n   */\r\n  async search(query: CaseSearchParams[]): Promise<CaseH[]> {\r\n    let filteredCases = await this.getCustomCases();\r\n    query.forEach(\r\n      ({ key, value }) =>\r\n        (filteredCases = filteredCases.filter((cs) =>\r\n          this.evaluateCaseValue(key, value, cs),\r\n        )),\r\n    );\r\n\r\n    return filteredCases;\r\n  }\r\n\r\n  /**\r\n   * Get all tissues in available cases.\r\n   * @param localization Language of tissue names.\r\n   */\r\n  async tissues(localization: string = 'EN'): Promise<CaseTissuesStains[]> {\r\n    if (!this.caseTissues) {\r\n      const cases = await this.getCustomCases();\r\n\r\n      const allTissues: CaseTissuesStains[] = [];\r\n      cases.forEach((c) =>\r\n        Object.entries(c.tissues)\r\n          .map(([tisName, tisValue]: [string, any]) => ({\r\n            name: tisName,\r\n            locName: tisValue[localization],\r\n          }))\r\n          .forEach((t) => allTissues.push(t)),\r\n      );\r\n      this.caseTissues = [\r\n        ...new Map(\r\n          allTissues.map((t) => [JSON.stringify([t.name, t.locName]), t]),\r\n        ).values(),\r\n      ];\r\n    }\r\n    return this.caseTissues;\r\n  }\r\n\r\n  /**\r\n   * Get all stains in available cases.\r\n   * @param localization Language of stains names.\r\n   */\r\n  async stains(localization: string = 'EN'): Promise<CaseTissuesStains[]> {\r\n    if (!this.caseStains) {\r\n      const cases = await this.getCustomCases();\r\n\r\n      const allStains: CaseTissuesStains[] = [];\r\n      cases.forEach((c) =>\r\n        Object.entries(c.stains)\r\n          .map(([stnName, stnValue]: [string, any]) => ({\r\n            name: stnName,\r\n            locName: stnValue[localization],\r\n          }))\r\n          .forEach((s) => allStains.push(s)),\r\n      );\r\n      this.caseStains = [\r\n        ...new Map(\r\n          allStains.map((s) => [JSON.stringify([s.name, s.locName]), s]),\r\n        ).values(),\r\n      ];\r\n    }\r\n    return this.caseStains;\r\n  }\r\n}\r\n","import Cases from '../root/cases';\r\nimport { Slide } from '../root/types/slide';\r\nimport { matchStringOnSeparatorGroup } from './utils';\r\nimport {AuthIntegration} from \"../integration\";\r\n\r\nexport default class WsiExplorer {\r\n  protected context: Cases;\r\n  protected lastCaseId: string | null = null;\r\n  protected data: Slide[] | null = null;\r\n  protected slidesData: Slide[] | null = null;\r\n  protected masksData: Slide[] | null = null;\r\n  protected integration: AuthIntegration;\r\n\r\n  maskIdentifierSeparator: string = '';\r\n  maskIdentifierValue: string = '';\r\n\r\n  constructor(context: Cases, integration: AuthIntegration) {\r\n    this.context = context;\r\n    this.integration = integration;\r\n  }\r\n\r\n  /**\r\n   * Configure WsiExplorer with regex identifying the local_id part and value this part should contain to distiguish masks and slides.\r\n   * @param maskIdentifierSeparator Regex specifying part of local_id that should identify the WSI type\r\n   * @param maskIdentifierValue Value the local_id part of WSI should have to identify the WSI as mask\r\n   */\r\n  use(maskIdentifierSeparator: string, maskIdentifierValue: string): void {\r\n    this.maskIdentifierSeparator = maskIdentifierSeparator;\r\n    this.maskIdentifierValue = maskIdentifierValue;\r\n  }\r\n\r\n  /**\r\n   * Fetch all WSIs of case\r\n   * @param caseId ID of case\r\n   */\r\n  private async getAllSlides(caseId: string): Promise<Slide[]> {\r\n    if (this.lastCaseId !== caseId || !this.data) {\r\n      this.data = (await this.context.slides(caseId)).items;\r\n    }\r\n    return this.data;\r\n  }\r\n\r\n  /**\r\n   * Fetch all actual slides of case\r\n   * @param caseId ID of case\r\n   */\r\n  async slides(caseId: string): Promise<Slide[]> {\r\n    if (this.lastCaseId !== caseId || !this.slidesData) {\r\n      this.slidesData = (await this.getAllSlides(caseId)).filter((slide) => {\r\n        return !matchStringOnSeparatorGroup(\r\n          slide.local_id || '',\r\n          this.maskIdentifierSeparator,\r\n          1,\r\n          this.maskIdentifierValue,\r\n        );\r\n      });\r\n    }\r\n    return this.slidesData;\r\n  }\r\n\r\n  /**\r\n   * Fetch all masks of a case\r\n   * @param caseId ID of case\r\n   */\r\n  async masks(caseId: string): Promise<Slide[]> {\r\n    if (this.lastCaseId !== caseId || !this.masksData) {\r\n      this.masksData = (await this.getAllSlides(caseId)).filter((slide) => {\r\n        return matchStringOnSeparatorGroup(\r\n          slide.local_id || '',\r\n          this.maskIdentifierSeparator,\r\n          1,\r\n          this.maskIdentifierValue,\r\n        );\r\n      });\r\n    }\r\n    return this.masksData;\r\n  }\r\n}\r\n","import { RootAPI, RootContext } from '../../root';\r\nimport CaseExplorer from '../extensions/case-explorer';\r\nimport WsiExplorer from '../extensions/wsi-explorer';\r\nimport Root from './root';\r\nimport { Case } from './types/case';\r\nimport { CaseList } from './types/case-list';\r\nimport { SlideList } from './types/slide-list';\r\n\r\nexport default class Cases extends RootContext {\r\n  protected context: RootAPI;\r\n  protected data: CaseList | null = null;\r\n\r\n  caseExplorer: CaseExplorer;\r\n  wsiExplorer: WsiExplorer;\r\n\r\n  constructor(context: Root) {\r\n    super();\r\n    this.context = context;\r\n\r\n    this.caseExplorer = new CaseExplorer(this, context.integration);\r\n    this.wsiExplorer = new WsiExplorer(this, context.integration);\r\n  }\r\n\r\n  async list(): Promise<CaseList> {\r\n    if (!this.data) {\r\n      this.data = (await this.context.rawQuery('/cases')) as CaseList;\r\n    }\r\n    return this.data;\r\n  }\r\n\r\n  async get(caseId: string): Promise<Case> {\r\n    return await this.context.rawQuery(`/cases/${caseId}`);\r\n  }\r\n\r\n  async slides(caseId: string): Promise<SlideList> {\r\n    return await this.context.rawQuery(`/cases/${caseId}/slides`);\r\n  }\r\n}\r\n","import { RootContext } from '../../root';\r\nimport Root from './root';\r\nimport { WorkbenchServiceApiV3CustomModelsExaminationsExamination } from './types/workbench-service-api-v-3-custom-models-examinations-examination';\r\nimport { ExaminationQuery } from './types/examination-query';\r\nimport { ExaminationList } from './types/examination-list';\r\nimport { ScopeTokenAndScopeId } from './types/scope-token-and-scope-id';\r\n\r\nexport default class Examinations extends RootContext {\r\n  protected context: Root;\r\n  protected data: ExaminationList | null = null;\r\n\r\n  constructor(context: Root) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  async create(\r\n    caseId: string,\r\n    appId: string,\r\n  ): Promise<WorkbenchServiceApiV3CustomModelsExaminationsExamination> {\r\n    const self = this.context;\r\n    self.requires('caseId', caseId);\r\n    return self.rawQuery('/examinations', {\r\n      method: 'PUT',\r\n      body: {\r\n        case_id: caseId,\r\n        app_id: appId,\r\n      },\r\n    });\r\n  }\r\n\r\n  async query(\r\n    query: ExaminationQuery,\r\n    skip?: number | undefined,\r\n    limit?: number | undefined,\r\n  ): Promise<ExaminationList> {\r\n    const self = this.context;\r\n    return self.rawQuery(`/examinations/query`, {\r\n      method: 'PUT',\r\n      body: query,\r\n      query: { skip, limit },\r\n    });\r\n  }\r\n\r\n  async get(\r\n    examinationId: string,\r\n  ): Promise<WorkbenchServiceApiV3CustomModelsExaminationsExamination> {\r\n    const self = this.context;\r\n    self.requires('examinationId', examinationId);\r\n    return self.rawQuery(`/examinations/${examinationId}`);\r\n  }\r\n\r\n  async scope(examinationId: string): Promise<ScopeTokenAndScopeId> {\r\n    const self = this.context;\r\n    self.requires('examinationId', examinationId);\r\n    return self.rawQuery(`/examinations/${examinationId}/scope`, {\r\n      method: 'PUT',\r\n    });\r\n  }\r\n}\r\n","import { ScopeContext } from '../../scope';\r\nimport Scope from './scope';\r\nimport { AppUiStorage } from './types/app-ui-storage';\r\n\r\nexport default class Storage extends ScopeContext {\r\n  protected context: Scope;\r\n  protected data: AppUiStorage | null = null;\r\n\r\n  constructor(context: Scope) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  async getRaw(): Promise<AppUiStorage> {\r\n    if (!this.data) {\r\n      this.data = await this.context.rawQuery('/app-ui-storage/user');\r\n    }\r\n    return this.data as AppUiStorage;\r\n  }\r\n\r\n  async flush(): Promise<AppUiStorage | null> {\r\n    if (!this.data) return null;\r\n\r\n    return (await this.context.rawQuery('/app-ui-storage/user', {\r\n      method: 'PUT',\r\n      body: this.data,\r\n    })) as AppUiStorage;\r\n  }\r\n\r\n  async get(key: string): Promise<any> {\r\n    const data = await this.getRaw();\r\n    if (typeof data.content[key] === 'string') {\r\n      return JSON.parse(data.content[key] as string);\r\n    }\r\n\r\n    return data.content[key];\r\n  }\r\n\r\n  async set(key: string, value: any, flush?: boolean): Promise<void> {\r\n    const valueRaw = JSON.stringify(value);\r\n    const dataRaw = await this.getRaw();\r\n    dataRaw.content[key] = valueRaw;\r\n    this.data = dataRaw;\r\n\r\n    if (flush) {\r\n      this.flush();\r\n    }\r\n  }\r\n\r\n  async erase(): Promise<void> {\r\n    this.data = { content: {} };\r\n    this.flush();\r\n  }\r\n}\r\n","import { ScopeContext } from '../../scope';\r\nimport Scope from './scope';\r\nimport { Annotation } from './types/annotation'\r\nimport { PostPointAnnotations } from './types/post-point-annotations';\r\nimport { PostLineAnnotations } from './types/post-line-annotations';\r\nimport { PostArrowAnnotations } from './types/post-arrow-annotations';\r\nimport { PostCircleAnnotations } from './types/post-circle-annotations';\r\nimport { PostRectangleAnnotations } from './types/post-rectangle-annotations';\r\nimport { PostPolygonAnnotations } from './types/post-polygon-annotations';\r\nimport { PostPointAnnotation } from './types/post-point-annotation';\r\nimport { PostLineAnnotation } from './types/post-line-annotation';\r\nimport { PostArrowAnnotation } from './types/post-arrow-annotation';\r\nimport { PostCircleAnnotation } from './types/post-circle-annotation';\r\nimport { PostRectangleAnnotation } from './types/post-rectangle-annotation';\r\nimport { PostPolygonAnnotation } from './types/post-polygon-annotation';\r\nimport { AnnotationListResponse } from './types/annotation-list-response';\r\nimport { IdObject } from './types/id-object';\r\nimport {AnnotationList} from \"./types/annotation-list\";\r\nimport {AnnotationQuery} from \"./types/annotation-query\";\r\nimport {ClassListResponse} from \"./types/class-list-response\";\r\nimport {Class} from \"./types/class\";\r\nimport {PostClassList} from \"./types/post-class-list\";\r\nimport {PostClass} from \"./types/post-class\";\r\nimport {ClassQuery} from \"./types/class-query\";\r\nimport {ClassList} from \"./types/class-list\";\r\n\r\nexport interface PostAnnotationQueryParams {\r\n  isRoi?: boolean;\r\n  externalIds?: boolean;\r\n}\r\n\r\nexport default class Annotations extends ScopeContext {\r\n  protected context: Scope;\r\n  protected data: undefined; //unused\r\n\r\n  constructor(context: Scope) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  /**\r\n   * Query Annotations\r\n   */\r\n  async query(data: AnnotationQuery, withClasses: boolean = true): Promise<AnnotationList> {\r\n    return await this.context.rawQuery('/annotations/query', {\r\n      method: 'PUT',\r\n      query: {\r\n        with_classes: withClasses\r\n      },\r\n      body: data\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Get Annotation by ID\r\n   */\r\n  async get(id: string, withClasses: boolean = true): Promise<Annotation> {\r\n    return await this.context.rawQuery(`/annotations/${id}`, {\r\n      query: {\r\n        with_classes: withClasses\r\n      }\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Post multiple annotations as one. In reality the same endpoint as create(...)\r\n   */\r\n  async createMany(\r\n    data:\r\n      | PostPointAnnotations\r\n      | PostLineAnnotations\r\n      | PostArrowAnnotations\r\n      | PostCircleAnnotations\r\n      | PostRectangleAnnotations\r\n      | PostPolygonAnnotations,\r\n    options: PostAnnotationQueryParams = {},\r\n  ): Promise<AnnotationListResponse> {\r\n    //the same as create but for types its simpler to split\r\n    return await this.context.rawQuery('/annotations', {\r\n      method: 'POST',\r\n      query: options,\r\n      body: data,\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Create an annotation.\r\n   */\r\n  async create(\r\n    data:\r\n      | PostPointAnnotation\r\n      | PostLineAnnotation\r\n      | PostArrowAnnotation\r\n      | PostCircleAnnotation\r\n      | PostRectangleAnnotation\r\n      | PostPolygonAnnotation,\r\n    options: PostAnnotationQueryParams = {},\r\n  ): Promise<Annotation> {\r\n    return await this.context.rawQuery('/annotations', {\r\n      method: 'POST',\r\n      query: options,\r\n      body: data,\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Delete an annotation. Does not remove related classes.\r\n   */\r\n  async deleteById(id: string): Promise<IdObject> {\r\n    return await this.context.rawQuery(`/annotations/${id}`, {\r\n      method: 'DELETE',\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Delete an annotation with its classes if defined.\r\n   */\r\n  async delete(object: Annotation): Promise<IdObject> {\r\n    if (!object.id) {\r\n      throw \"Cannot delete annotation without ID property!\";\r\n    }\r\n    const deleted = await this.deleteById(object.id);\r\n    if (object.classes) {\r\n      for (let cls of object.classes) {\r\n        if (cls.id) await this.deleteClass(cls.id);\r\n      }\r\n    }\r\n    return deleted;\r\n  }\r\n\r\n  /**\r\n   * Update annotation: delete and re-create\r\n   */\r\n  async update(\r\n    id: string,\r\n    data:\r\n      | PostPointAnnotation\r\n      | PostLineAnnotation\r\n      | PostArrowAnnotation\r\n      | PostCircleAnnotation\r\n      | PostRectangleAnnotation\r\n      | PostPolygonAnnotation,\r\n    options: PostAnnotationQueryParams = {},\r\n  ): Promise<Annotation> {\r\n    await this.deleteById(id);\r\n\r\n    // update might carry id but user forgot to set external IDs to true\r\n    if (!options.externalIds && data.id) {\r\n      options.externalIds = true;\r\n    }\r\n    return await this.create(data, options);\r\n  }\r\n\r\n  /**\r\n   * Attach class to an existing annotation\r\n   */\r\n  async addClass(data: PostClass): Promise<Class> {\r\n    return await this.context.rawQuery('/classes', {\r\n      method: 'POST',\r\n      body: data,\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Attach classes to existing annotations\r\n   */\r\n  async addClassMany(data: PostClassList): Promise<ClassListResponse> {\r\n    return await this.context.rawQuery('/classes', {\r\n      method: 'POST',\r\n      body: data,\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Get class by its ID\r\n   */\r\n  async getClass(id: string): Promise<Class> {\r\n    return await this.context.rawQuery(`/classes/${id}`);\r\n  }\r\n\r\n  /**\r\n   * Delete class by its ID\r\n   */\r\n  async deleteClass(id: string): Promise<IdObject> {\r\n    return await this.context.rawQuery(`/classes/${id}`, {\r\n      method: 'DELETE',\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Query classes by properties\r\n   */\r\n  async queryClasses(data: ClassQuery): Promise<ClassList> {\r\n    return await this.context.rawQuery('/classes/query', {\r\n      method: 'PUT',\r\n      body: data\r\n    });\r\n  }\r\n}\r\n","import { ScopeContext } from '../../scope';\r\nimport Scope from './scope';\r\nimport { Collection } from './types/collection';\r\nimport { ItemQuery } from './types/item-query';\r\nimport { ItemQueryList } from './types/item-query-list';\r\n\r\nexport default class Collections extends ScopeContext {\r\n  protected context: Scope;\r\n  protected data: Collection | null = null;\r\n\r\n  constructor(context: Scope) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  async get(collectionId: string): Promise<Collection> {\r\n    const collection: Collection = await this.context.rawQuery(\r\n      `/collections/${collectionId}`,\r\n    );\r\n\r\n    return collection;\r\n  }\r\n\r\n  async create(collection: Collection): Promise<Collection> {\r\n    const createdCollection: Collection = await this.context.rawQuery(\r\n      `/collections`,\r\n      {\r\n        method: 'POST',\r\n        body: collection,\r\n      },\r\n    );\r\n\r\n    return createdCollection;\r\n  }\r\n\r\n  async delete(collectionId: string): Promise<void> {\r\n    await this.context.rawQuery(`/collections/${collectionId}`, {\r\n      method: 'DELETE',\r\n    });\r\n  }\r\n\r\n  async queryItems(\r\n    collectionId: string,\r\n    query: ItemQuery,\r\n  ): Promise<ItemQueryList> {\r\n    const queryResult: ItemQueryList = await this.context.rawQuery(\r\n      `/collections/${collectionId}/items/query`,\r\n      {\r\n        method: 'PUT',\r\n        body: query,\r\n      },\r\n    );\r\n\r\n    return queryResult;\r\n  }\r\n\r\n  async createItems(collectionId: string, items: any): Promise<void> {\r\n    await this.context.rawQuery(`/collections/${collectionId}/items`, {\r\n      method: 'POST',\r\n      body: {\r\n        ...items,\r\n      },\r\n    });\r\n  }\r\n\r\n  async deleteItem(collectionId: string, itemId: string): Promise<void> {\r\n    await this.context.rawQuery(\r\n      `/collections/${collectionId}/items/${itemId}`,\r\n      {\r\n        method: 'DELETE',\r\n      },\r\n    );\r\n  }\r\n}\r\n","import { ScopeContext } from '../../scope';\r\nimport Scope from './scope';\r\nimport { Job } from './types/job';\r\nimport { JobList } from './types/job-list';\r\n\r\nexport default class Jobs extends ScopeContext {\r\n  protected context: Scope;\r\n  protected data: Job[] | null = null;\r\n\r\n  constructor(context: Scope) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  async getJobs(): Promise<Job[]> {\r\n    return ((await this.context.rawQuery('/jobs')) as JobList).items;\r\n  }\r\n\r\n  async get(jobId: string): Promise<Job> {\r\n    return (await this.context.rawQuery(`/jobs/${jobId}`)) as Job;\r\n  }\r\n}\r\n","import { ScopeContext } from '../../scope';\r\nimport Scope from './scope';\r\nimport { ContinuousPixelmap } from './types/continuous-pixelmap';\r\nimport { DiscretePixelmap } from './types/discrete-pixelmap';\r\nimport { IdObject } from './types/id-object';\r\nimport { NominalPixelmap } from './types/nominal-pixelmap';\r\nimport { PixelmapList } from './types/pixelmap-list';\r\nimport { PixelmapQuery } from './types/pixelmap-query';\r\n\r\ntype Pixelmap = ContinuousPixelmap | DiscretePixelmap | NominalPixelmap;\r\n\r\nexport default class Pixelmaps extends ScopeContext {\r\n  protected context: Scope;\r\n  protected data: Pixelmap | null = null;\r\n\r\n  constructor(context: Scope) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  async get(pixelmapId: string): Promise<Pixelmap> {\r\n    return (await this.context.rawQuery(\r\n      `/pixelmaps/${pixelmapId}`,\r\n    )) as Pixelmap;\r\n  }\r\n\r\n  async post(\r\n    pixelmaps: Pixelmaps,\r\n  ): Promise<\r\n    ContinuousPixelmap | DiscretePixelmap | NominalPixelmap | PixelmapList\r\n  > {\r\n    return await this.context.rawQuery(`/pixelmaps`, {\r\n      method: 'POST',\r\n      body: pixelmaps,\r\n    });\r\n  }\r\n\r\n  async delete(pixelmapId: string): Promise<IdObject> {\r\n    return (await this.context.rawQuery(`/pixelmaps/${pixelmapId}`, {\r\n      method: 'DELETE',\r\n    })) as IdObject;\r\n  }\r\n\r\n  async query(query: PixelmapQuery): Promise<Pixelmap[]> {\r\n    const queryResult = (await this.context.rawQuery(`/pixelmaps/query`, {\r\n      method: 'PUT',\r\n      body: query,\r\n    })) as PixelmapList;\r\n    return queryResult.items;\r\n  }\r\n\r\n  async getTile(\r\n    pixelmapId: string,\r\n    level: number,\r\n    tileX: number,\r\n    tileY: number,\r\n  ): Promise<Blob> {\r\n    const tile = await this.context.rawQuery(\r\n      `/pixelmaps/${pixelmapId}/level/${level}/position/${tileX}/${tileY}/data`,\r\n      {\r\n        responseType: 'blob',\r\n      },\r\n    );\r\n    return tile;\r\n  }\r\n\r\n  async uploadTile(\r\n    pixelmapId: string,\r\n    level: number,\r\n    tileX: number,\r\n    tileY: number,\r\n    tile: Blob,\r\n  ): Promise<string> {\r\n    const result = await this.context.rawQuery(\r\n      `/pixelmaps/${pixelmapId}/level/${level}/position/${tileX}/${tileY}/data`,\r\n      {\r\n        method: 'PUT',\r\n        body: tile,\r\n      },\r\n    );\r\n    return result;\r\n  }\r\n\r\n  async deleteTile(\r\n    pixelmapId: string,\r\n    level: number,\r\n    tileX: number,\r\n    tileY: number,\r\n  ): Promise<string> {\r\n    const result = await this.context.rawQuery(\r\n      `/pixelmaps/${pixelmapId}/level/${level}/position/${tileX}/${tileY}/data`,\r\n      {\r\n        method: 'DELETE',\r\n      },\r\n    );\r\n    return result;\r\n  }\r\n\r\n  async bulkGetTile(\r\n    pixelmapId: string,\r\n    level: number,\r\n    startX: number,\r\n    startY: number,\r\n    endX: number,\r\n    endY: number,\r\n  ): Promise<Blob> {\r\n    const tiles = await this.context.rawQuery(\r\n      `/pixelmaps/${pixelmapId}/level/${level}/position/start/${startX}/${startY}/end/${endX}/${endY}/data`,\r\n      {\r\n        responseType: 'blob',\r\n      },\r\n    );\r\n    return tiles;\r\n  }\r\n\r\n  async bulkUploadTile(\r\n    pixelmapId: string,\r\n    level: number,\r\n    startX: number,\r\n    startY: number,\r\n    endX: number,\r\n    endY: number,\r\n    tiles: Blob,\r\n  ): Promise<string> {\r\n    const result = await this.context.rawQuery(\r\n      `/pixelmaps/${pixelmapId}/level/${level}/position/start/${startX}/${startY}/end/${endX}/${endY}/data`,\r\n      {\r\n        method: 'PUT',\r\n        body: tiles,\r\n      },\r\n    );\r\n    return result;\r\n  }\r\n}\r\n","import { ScopeAPI } from '../../scope';\r\nimport { RawAPI, RawOptions } from '../../base';\r\nimport { ScopeTokenAndScopeId } from '../root/types/scope-token-and-scope-id';\r\nimport { WorkbenchServiceApiV3CustomModelsExaminationsExamination } from '../root/types/workbench-service-api-v-3-custom-models-examinations-examination';\r\nimport Root from '../root/root';\r\nimport Storage from './storage';\r\nimport {\r\n  getJwtTokenExpiresTimeout,\r\n  parseJwtToken,\r\n  ScopeToken,\r\n} from '../../utils';\r\nimport Annotations from './annotations';\r\nimport Collections from './collections';\r\nimport Jobs from './jobs';\r\nimport Pixelmaps from \"./pixelmaps\";\r\n\r\nexport default class Scope extends ScopeAPI {\r\n  static apiPath = '/v3/scopes';\r\n\r\n  // Interface\r\n  raw: RawAPI;\r\n  context: Root;\r\n  storage: Storage;\r\n  annotations: Annotations;\r\n  collections: Collections;\r\n  jobs: Jobs;\r\n  pixelmaps: Pixelmaps;\r\n\r\n  // Additional\r\n  scopeContext: ScopeTokenAndScopeId | null = null;\r\n  private _defaultExaminationId: string = '';\r\n  private _tokenRefetchInterval: NodeJS.Timeout | null = null;\r\n\r\n  activeExaminationId: string = '';\r\n  activeCaseId: string = '';\r\n  activeAppId: string = '';\r\n\r\n  constructor(context: Root) {\r\n    super();\r\n    this.context = context;\r\n    this.raw = new RawAPI(this.context.options.apiUrl + Scope.apiPath);\r\n    this.storage = new Storage(this);\r\n    this.annotations = new Annotations(this);\r\n    this.collections = new Collections(this);\r\n    this.jobs = new Jobs(this);\r\n    this.pixelmaps = new Pixelmaps(this);\r\n  }\r\n\r\n  async use(\r\n    caseId: string,\r\n    appId: string | undefined = undefined,\r\n  ): Promise<void> {\r\n    //todo consider caching\r\n\r\n    this.requires('root::userId', this.context.userId);\r\n\r\n    const findExamination = async (\r\n      appId: string,\r\n    ): Promise<WorkbenchServiceApiV3CustomModelsExaminationsExamination> => {\r\n      let examinations = await this.context.examinations.query({\r\n        apps: [appId],\r\n        creators: [this.context.userId],\r\n      });\r\n      if (examinations.item_count > 0) {\r\n        let examination = examinations.items.find((ex) => ex.state === 'OPEN');\r\n        if (examination) return examination;\r\n      }\r\n      return await this.context.examinations.create(caseId, appId);\r\n    };\r\n\r\n    let examination;\r\n    if (appId) {\r\n      examination = await findExamination(appId);\r\n    } else if (this._defaultExaminationId) {\r\n      examination = await this.context.examinations.get(\r\n        this._defaultExaminationId,\r\n      );\r\n    }\r\n\r\n    if (!examination) {\r\n      let app = await this.context.apps.default();\r\n      examination = await findExamination(app.app_id);\r\n      this._defaultExaminationId = examination.id;\r\n    }\r\n\r\n    await this.from(examination);\r\n  }\r\n\r\n  get scopeToken(): string {\r\n    return this.scopeContext?.access_token || \"\";\r\n  }\r\n\r\n  get id(): string {\r\n    return this.scopeContext?.scope_id || \"\";\r\n  }\r\n\r\n  async from(\r\n    examination: WorkbenchServiceApiV3CustomModelsExaminationsExamination,\r\n  ): Promise<void> {\r\n    this.reset();\r\n    this.scopeContext = await this.context.examinations.scope(examination.id);\r\n    this.activeCaseId = examination.case_id;\r\n    this.activeAppId = examination.app_id;\r\n    this.activeExaminationId = examination.id;\r\n    const token = parseJwtToken(this.scopeContext.access_token) as ScopeToken;\r\n    const timeout = getJwtTokenExpiresTimeout(token);\r\n    this._tokenRefetchInterval = setInterval(async () => {\r\n      this.scopeContext = await this.context.examinations.scope(examination.id);\r\n    }, timeout);\r\n    this.raiseEvent('init');\r\n  }\r\n\r\n  reset(): void {\r\n    this.activeExaminationId = '';\r\n    this.scopeContext = null;\r\n    if (this._tokenRefetchInterval) {\r\n      clearInterval(this._tokenRefetchInterval);\r\n      this._tokenRefetchInterval = null;\r\n      //todo clear all cached data\r\n      this.raiseEvent('reset');\r\n    }\r\n  }\r\n\r\n  async rawQuery(endpoint: string, options?: RawOptions): Promise<any> {\r\n    this.requires('this.scopeContext', this.scopeContext);\r\n    options = options || {};\r\n    options.headers = options.headers || {};\r\n    options.headers['Authorization'] =\r\n      `Bearer ${this.scopeContext?.access_token}`;\r\n    if (endpoint && !endpoint.startsWith('/')) {\r\n      endpoint = `/${endpoint}`;\r\n    }\r\n\r\n    try {\r\n      return await this.raw.http(\r\n        `/${this.scopeContext?.scope_id}${endpoint}`,\r\n        options,\r\n      );\r\n    } catch (e) {\r\n      if (e.statusCode === 401) {\r\n        this.scopeContext = await this.context.examinations.scope(\r\n          this.activeExaminationId,\r\n        );\r\n        return await this.raw.http(\r\n          `/${this.scopeContext.scope_id}${endpoint}`,\r\n          options,\r\n        );\r\n      }\r\n\r\n      throw e;\r\n    }\r\n  }\r\n}\r\n","/* tslint:disable */\r\n/* eslint-disable */\r\nexport enum GlobalDataCreatorType {\r\n  USER = 'user',\r\n  SCOPE = 'scope',\r\n  JOB = 'job',\r\n}\r\n","/* tslint:disable */\r\n/* eslint-disable */\r\nexport enum GlobalItemReferenceType {\r\n  ANNOTATION = 'annotation',\r\n  COLLECTION = 'collection',\r\n  CLASS = 'class',\r\n  PRIMITIVE = 'primitive',\r\n  WSI = 'wsi',\r\n  CASE = 'case',\r\n  USER = 'user',\r\n  SCOPE = 'scope',\r\n  JOB = 'job',\r\n}\r\n","/* tslint:disable */\r\n/* eslint-disable */\r\nexport enum TemplateType {\r\n  Background = 'background',\r\n  Params = 'params',\r\n  Shader = 'shader',\r\n  Visualization = 'visualization',\r\n}\r\n","import { RootAPI, RootContext } from '../../root';\r\nimport Root from './root';\r\nimport { Slide } from './types/slide';\r\nimport { SlideInfo } from './types/slide-info';\r\n\r\nexport default class Slides extends RootContext {\r\n  protected context: RootAPI;\r\n  protected data: SlideInfo | null = null;\r\n\r\n  constructor(context: Root) {\r\n    super();\r\n    this.context = context;\r\n  }\r\n\r\n  async slideInfo(slideId: string): Promise<SlideInfo> {\r\n    return await this.context.rawQuery(`/slides/${slideId}/info`);\r\n  }\r\n\r\n  async slideThumbnail(\r\n    slideId: string,\r\n    maxWidth: number,\r\n    maxHeight: number,\r\n    format?: string,\r\n  ): Promise<Blob> {\r\n    return await this.context.rawQuery(\r\n      `/slides/${slideId}/thumbnail/max_size/${maxWidth}/${maxHeight}`,\r\n      {\r\n        query: {\r\n          image_format: format,\r\n        },\r\n        responseType: 'blob',\r\n      },\r\n    );\r\n  }\r\n\r\n  async slideLabel(\r\n    slideId: string,\r\n    maxWidth: number,\r\n    maxHeight: number,\r\n    format?: string,\r\n  ): Promise<Blob> {\r\n    return await this.context.rawQuery(\r\n      `/slides/${slideId}/label/max_size/${maxWidth}/${maxHeight}`,\r\n      {\r\n        query: {\r\n          image_format: format,\r\n        },\r\n        responseType: 'blob',\r\n      },\r\n    );\r\n  }\r\n\r\n  async loadTile(\r\n    slideId: string,\r\n    level: number,\r\n    x: number,\r\n    y: number,\r\n    format?: string,\r\n  ): Promise<Blob> {\r\n    return await this.context.rawQuery(\r\n      `/slides/${slideId}/tile/level/${level}/tile/${x}/${y}`,\r\n      {\r\n        query: {\r\n          image_format: format,\r\n        },\r\n        responseType: 'blob',\r\n      },\r\n    );\r\n  }\r\n}\r\n","import GlobalStorage from '../rationai/global-storage';\r\nimport { GlobalItem } from '../rationai/types/global-item';\r\nimport { GlobalItemReferenceType } from '../rationai/types/global-item-reference-type';\r\nimport { MaskMetadata } from './types/mask-metadata';\r\nimport { Shader, SlideMetadata } from './types/slide-metadata';\r\nimport { TemplateType } from './types/template-type';\r\n\r\nexport default class WsiMetadata {\r\n  protected context: GlobalStorage;\r\n  protected data: GlobalItem | null = null;\r\n\r\n  private defaultSlideMetadata: SlideMetadata = { visualization: {} };\r\n  private defaultMaskMetadata: MaskMetadata = {};\r\n\r\n  constructor(context: GlobalStorage) {\r\n    this.context = context;\r\n  }\r\n\r\n  // Slide\r\n\r\n  /**\r\n   * Fetch global item containing metadata of WSI.\r\n   * @param wsiId ID of WSI\r\n   * @param isSlide Specify if WSI is a slide or a mask\r\n   */\r\n  private async getWsiMetadataItem(\r\n    wsiId: string,\r\n    isSlide: boolean = true,\r\n  ): Promise<GlobalItem> {\r\n    let metadata = (await this.context.query({ references: [wsiId] })).find(\r\n      (p) => p.data_type === `${isSlide ? 'slide' : 'mask'}_metadata`,\r\n    );\r\n    if (!metadata) {\r\n      metadata = await this.createWsiMetadataItem(\r\n        wsiId,\r\n        isSlide ? this.defaultSlideMetadata : this.defaultMaskMetadata,\r\n      );\r\n    }\r\n    return metadata;\r\n  }\r\n\r\n  /**\r\n   * Create global item containing metadata of WSI.\r\n   * @param wsiId ID of WSI\r\n   * @param value Metadata of WSI\r\n   * @param isSlide Specify if WSI is a slide or a mask\r\n   */\r\n  private async createWsiMetadataItem(\r\n    wsiId: string,\r\n    value: any,\r\n    isSlide: boolean = true,\r\n  ): Promise<GlobalItem> {\r\n    return await this.context.createValue(\r\n      value,\r\n      `Metadata of ${isSlide ? 'slide' : 'mask'} ${wsiId}`,\r\n      undefined,\r\n      wsiId,\r\n      GlobalItemReferenceType.WSI,\r\n      `${isSlide ? 'slide' : 'mask'}_metadata`,\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Get metadata of slide WSI.\r\n   * @param slideId ID of slide WSI\r\n   */\r\n  async getSlideMetadata(slideId: string): Promise<SlideMetadata> {\r\n    return JSON.parse((await this.getWsiMetadataItem(slideId)).value as string);\r\n  }\r\n\r\n  /**\r\n   * Update metadata of slide WSI.\r\n   * @param slideId ID of slide WSI\r\n   * @param value New metadata of WSI\r\n   */\r\n  async updateSlideMetadata(\r\n    slideId: string,\r\n    value: SlideMetadata,\r\n  ): Promise<SlideMetadata | false> {\r\n    const metadataItem = await this.getWsiMetadataItem(slideId);\r\n    try {\r\n      const updatedItem = await this.context.update(metadataItem.id, {\r\n        ...metadataItem,\r\n        value: JSON.stringify(value),\r\n      });\r\n      return JSON.parse(updatedItem.value);\r\n    } catch (e) {\r\n      return false;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Helper method generating shader config for user-created xOpat visualisations\r\n   * @param shaders Array of shaders\r\n   */\r\n  private async getShadersConfig(shaders: Shader[]): Promise<object> {\r\n    const shadersConfig = {};\r\n    for (let i = 0; i < shaders.length; i++) {\r\n      const shaderTemp = await this.context.visTemplates.getTemplate(\r\n        TemplateType.Shader,\r\n        shaders[i].shaderTemplate,\r\n      );\r\n      shadersConfig[shaders[i].id] = {\r\n        ...shaderTemp,\r\n        name: shaders[i].name || shaders[i].id,\r\n        dataReferences: shaders[i].dataRefs,\r\n      };\r\n    }\r\n    return shadersConfig;\r\n  }\r\n\r\n  /**\r\n   * Fetch user-created xOpat visualisations of a slide\r\n   * @param slideId ID of a slide\r\n   */\r\n  async getVisualizations(slideId: string): Promise<any> {\r\n    const slideVis = (await this.getSlideMetadata(slideId)).visualization;\r\n    const visualizations = {\r\n      params: slideVis.paramsTemplate\r\n        ? (await this.context.visTemplates.getTemplate(\r\n            TemplateType.Params,\r\n            slideVis.paramsTemplate,\r\n          )) || undefined\r\n        : undefined,\r\n      data: slideVis.data,\r\n      background: {\r\n        ...(slideVis.background\r\n          ? await this.context.visTemplates.getTemplate(\r\n              TemplateType.Background,\r\n              slideVis.background?.template,\r\n            )\r\n          : {}),\r\n        data: slideVis.background && slideVis.background.dataRef,\r\n      },\r\n      visualizations:\r\n        slideVis.visualizations &&\r\n        (await Promise.all(\r\n          slideVis.visualizations.map(async (vis) => {\r\n            return {\r\n              ...(await this.context.visTemplates.getTemplate(\r\n                TemplateType.Visualization,\r\n                vis.visTemplate,\r\n              )),\r\n              name: vis.name,\r\n              shaders: await this.getShadersConfig(vis.shaders),\r\n            };\r\n          }),\r\n        )),\r\n    };\r\n    return visualizations;\r\n  }\r\n\r\n  // MASK\r\n\r\n  /**\r\n   * Get metadata of mask WSI.\r\n   * @param slideId ID of mask WSI\r\n   */\r\n  async getMaskMetadata(maskId: string): Promise<MaskMetadata> {\r\n    return JSON.parse(\r\n      (await this.getWsiMetadataItem(maskId, false)).value as string,\r\n    );\r\n  }\r\n}\r\n","import GlobalStorage from '../rationai/global-storage';\r\nimport { GlobalItem } from '../rationai/types/global-item';\r\nimport { TemplateType } from './types/template-type';\r\n\r\nconst templatesGlobalItemDataType = 'vis_templates';\r\n\r\nexport default class VisualizationTemplates {\r\n  protected context: GlobalStorage;\r\n  protected data: object | null = null;\r\n\r\n  constructor(context: GlobalStorage) {\r\n    this.context = context;\r\n  }\r\n\r\n  /**\r\n   * Fetch global item containing template.\r\n   * @param type Type of the template (backgroung, params, shader, visualization)\r\n   * @param name Name of the template\r\n   */\r\n  async fetchTemplateItem(\r\n    type: TemplateType,\r\n    name: string,\r\n  ): Promise<GlobalItem | undefined> {\r\n    return (\r\n      await this.context.query({\r\n        references: [null],\r\n        data_types: [`${templatesGlobalItemDataType}_${type}`],\r\n      })\r\n    ).find((item) => item.name === name);\r\n  }\r\n\r\n  /**\r\n   * Fetch template.\r\n   * @param type Type of the template (backgroung, params, shader, visualization)\r\n   * @param name Name of the template\r\n   */\r\n  async getTemplate(type: TemplateType, name: string): Promise<object | false> {\r\n    const tmpl = await this.fetchTemplateItem(type, name);\r\n    if (tmpl) {\r\n      return JSON.parse(tmpl.value);\r\n    }\r\n    return false;\r\n  }\r\n\r\n  /**\r\n   * Create new template.\r\n   * @param type Type of the template (background, params, shader, visualization)\r\n   * @param name Name of the template\r\n   * @param value Value of the template\r\n   */\r\n  async createTemplate(\r\n    type: TemplateType,\r\n    name: string,\r\n    value: object,\r\n  ): Promise<object | false> {\r\n    const existingTmpl = await this.fetchTemplateItem(type, name);\r\n    if (existingTmpl) {\r\n      return false;\r\n    }\r\n    return await this.context.createValue(\r\n      value,\r\n      `${name}`,\r\n      undefined,\r\n      undefined,\r\n      undefined,\r\n      `${templatesGlobalItemDataType}_${type}`,\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Delete template.\r\n   * @param type Type of the template (backgroung, params, shader, visualization)\r\n   * @param name Name of the template\r\n   */\r\n  async deleteTemplate(type: TemplateType, name: string): Promise<boolean> {\r\n    const existingTmpl = await this.fetchTemplateItem(type, name);\r\n    if (!existingTmpl) {\r\n      return false;\r\n    }\r\n    await this.context.delete(existingTmpl.id);\r\n    return true;\r\n  }\r\n}\r\n","import { HTTPError } from '../../../src/base';\r\nimport GlobalStorage from '../rationai/global-storage';\r\nimport { GlobalItem } from '../rationai/types/global-item';\r\nimport { AnnotPreset, AnnotPresetObject } from './types/annot-preset';\r\n\r\ntype AnnotPresetGetResult = {\r\n  presets: AnnotPreset[];\r\n  lastModifiedAt: number;\r\n};\r\n\r\ntype AnnotPresetUpdateResult = {\r\n  presets: AnnotPreset[];\r\n  successfulUpdate: boolean;\r\n  lastModifiedAt: number;\r\n};\r\n\r\nexport default class AnnotPresets {\r\n  protected context: GlobalStorage;\r\n  protected data: GlobalItem | null = null;\r\n\r\n  protected presetDataType: string = 'annot_presets';\r\n\r\n  constructor(context: GlobalStorage) {\r\n    this.context = context;\r\n  }\r\n\r\n  /**\r\n   * Configure AnnotPresets class with data type. Data type is used to filter global items.\r\n   * @param presetDataType Data type of global items used to store annotation presets.\r\n   */\r\n  use(presetDataType: string): void {\r\n    this.presetDataType = presetDataType;\r\n  }\r\n\r\n  /**\r\n   * Fetch global item containing annotation presets (only one should exist).\r\n   * @param fresh Force fresh fetch of global item, otherwise cached version might be used\r\n   */\r\n  private async getPresetsItem(fresh: boolean = false): Promise<GlobalItem> {\r\n    if (!this.data || fresh) {\r\n      let presetsItem = (\r\n        await this.context.query({\r\n          references: [null],\r\n          data_types: [this.presetDataType],\r\n        })\r\n      ).find((item) => true);\r\n      if (!presetsItem) {\r\n        presetsItem = await this.createPresetsItem({ presets: [] });\r\n      }\r\n      this.data = presetsItem;\r\n    }\r\n    return this.data;\r\n  }\r\n\r\n  /**\r\n   * Create global item containing annotation presets (only one should exist).\r\n   * @param value Annotation preset\r\n   */\r\n  private async createPresetsItem(\r\n    value: AnnotPresetObject,\r\n  ): Promise<GlobalItem> {\r\n    return await this.context.createValue(\r\n      value,\r\n      `Global annotation presets`,\r\n      undefined,\r\n      undefined,\r\n      undefined,\r\n      this.presetDataType,\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Get annotation presets.\r\n   * @param fresh Force fresh fetch\r\n   */\r\n  async getAnnotPresets(fresh: boolean = false): Promise<AnnotPresetGetResult> {\r\n    const presetItem = await this.getPresetsItem(fresh);\r\n    return {\r\n      presets: (JSON.parse(presetItem.value as string) as AnnotPresetObject)\r\n        .presets,\r\n      lastModifiedAt: presetItem.modified_at,\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Helper function to merge annotation presets during parallel updates.\r\n   */\r\n  private mergePresets(\r\n    primaryArr: AnnotPreset[],\r\n    secondaryArr: AnnotPreset[],\r\n    localVersion: number,\r\n  ): AnnotPreset[] {\r\n    const newArr = [...primaryArr];\r\n    // item from secondary array is pushed only if item with the same id is not present in primary array, and createdAt date is bigger than localVersion date, meaning item is new\r\n    secondaryArr.forEach((secItem) =>\r\n      newArr.some((primItem) => primItem.id === secItem.id) ||\r\n      !secItem.createdAt ||\r\n      secItem.createdAt <= localVersion\r\n        ? null\r\n        : newArr.push(secItem),\r\n    );\r\n    return newArr;\r\n  }\r\n\r\n  /**\r\n   * Update annotation presets.\r\n   * @param value New presets\r\n   * @param localVersion Local version of presets (modified_at attribute of global item)\r\n   * @param failOnParallelUpdate Force update fail if local version is outdated\r\n   */\r\n  async updateAnnotPresets(\r\n    value: AnnotPreset[],\r\n    localVersion: number,\r\n    failOnParallelUpdate: boolean = false,\r\n  ): Promise<AnnotPresetUpdateResult> {\r\n    // fetch fresh presets\r\n    const remotePresetsItem = await this.getPresetsItem(true);\r\n    const remotePresets = (\r\n      JSON.parse(remotePresetsItem.value as string) as AnnotPresetObject\r\n    ).presets;\r\n\r\n    let localPresets = value;\r\n    let successfulUpdate = true;\r\n\r\n    if (remotePresetsItem.modified_at !== localVersion) {\r\n      if (failOnParallelUpdate) {\r\n        return {\r\n          presets: remotePresets,\r\n          successfulUpdate: false,\r\n          lastModifiedAt: remotePresetsItem.modified_at,\r\n        };\r\n      }\r\n      localPresets = this.mergePresets(\r\n        remotePresets,\r\n        localPresets,\r\n        localVersion,\r\n      );\r\n      successfulUpdate = false;\r\n    }\r\n\r\n    try {\r\n      const updatedItem = await this.context.update(remotePresetsItem.id, {\r\n        ...remotePresetsItem,\r\n        value: JSON.stringify({ presets: localPresets }),\r\n      });\r\n      const updatedPresets = (\r\n        JSON.parse(updatedItem.value) as AnnotPresetObject\r\n      ).presets;\r\n      return {\r\n        presets: updatedPresets,\r\n        successfulUpdate: successfulUpdate,\r\n        lastModifiedAt: updatedItem.modified_at,\r\n      };\r\n    } catch (e) {\r\n      if ((e as HTTPError).statusCode === 409) {\r\n        const retryAttempt = await this.updateAnnotPresets(\r\n          localPresets,\r\n          remotePresetsItem.modified_at,\r\n        );\r\n        return { ...retryAttempt, successfulUpdate: successfulUpdate };\r\n      }\r\n      throw e;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Delete annotation presets.\r\n   */\r\n  async deleteAnnotPresets(): Promise<void> {\r\n    const presetsItem = await this.getPresetsItem(true);\r\n    await this.context.delete(presetsItem.id);\r\n    this.data = null;\r\n  }\r\n}\r\n","import GlobalStorage from '../rationai/global-storage';\r\nimport { GlobalItem } from '../rationai/types/global-item';\r\nimport { GlobalItemReferenceType } from '../rationai/types/global-item-reference-type';\r\n\r\nexport default class JobConfig {\r\n  protected context: GlobalStorage;\r\n  protected data: object | null = null;\r\n\r\n  protected configDataType: string = 'app_job_config';\r\n\r\n  constructor(context: GlobalStorage) {\r\n    this.context = context;\r\n  }\r\n\r\n  /**\r\n   * Configure JobConfig class with data type. Data type is used to filter global items.\r\n   * @param configDataType Data type of global items used to store job configs.\r\n   */\r\n  use(configDataType: string): void {\r\n    this.configDataType = configDataType;\r\n  }\r\n\r\n  /**\r\n   * Fetch global item containing job config of an App.\r\n   * @param appId ID of App\r\n   */\r\n  private async fetchJobConfigItem(\r\n    appId: string,\r\n  ): Promise<GlobalItem | undefined> {\r\n    return (\r\n      await this.context.query({\r\n        references: [appId],\r\n        data_types: [this.configDataType],\r\n      })\r\n    ).find(Boolean);\r\n  }\r\n\r\n  /**\r\n   * Get job config of an App.\r\n   * @param appId ID of App\r\n   */\r\n  async getJobConfig(appId: string): Promise<object | false> {\r\n    const item = await this.fetchJobConfigItem(appId);\r\n    if (item) {\r\n      return JSON.parse(item.value);\r\n    }\r\n    return false;\r\n  }\r\n\r\n  /**\r\n   * Get job config of an App.\r\n   * @param appId ID of App\r\n   * @param value Job config of an App\r\n   */\r\n  async createJobConfig(appId: string, value: object): Promise<object | false> {\r\n    const existingConfig = await this.fetchJobConfigItem(appId);\r\n    if (existingConfig) {\r\n      return false;\r\n    }\r\n    return await this.context.createValue(\r\n      value,\r\n      `Job config of App`,\r\n      undefined,\r\n      appId,\r\n      GlobalItemReferenceType.JOB,\r\n      this.configDataType,\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Delete job config of an App.\r\n   * @param appId ID of App\r\n   */\r\n  async deleteJobConfig(appId: string): Promise<boolean> {\r\n    const existingConfig = await this.fetchJobConfigItem(appId);\r\n    if (!existingConfig) {\r\n      return false;\r\n    }\r\n    await this.context.delete(existingConfig.id);\r\n    return true;\r\n  }\r\n}\r\n","import RationAI from './rationai';\r\nimport { GlobalItem } from './types/global-item';\r\nimport { GlobalStorageQuery } from './types/global-storage-query';\r\nimport { GlobalItemList } from './types/global-item-list';\r\nimport { PostGlobalItems } from './types/post-global-items';\r\nimport { GlobalItems } from './types/global-items';\r\nimport { GlobalItemReferenceType } from './types/global-item-reference-type';\r\nimport { PostGlobalItem } from './types/post-global-item';\r\nimport { GlobalDataCreatorType } from './types/global-data-creator-type';\r\nimport { PutGlobalItem } from './types/put-global-item';\r\nimport WsiMetadata from '../extensions/wsi-metadata';\r\nimport VisualizationTemplates from '../extensions/visualization-templates';\r\nimport AnnotPresets from '../extensions/annot-presets';\r\nimport JobConfig from '../extensions/job-config';\r\n\r\nexport default class GlobalStorage {\r\n  protected context: RationAI;\r\n  protected data: GlobalItem | null = null;\r\n\r\n  wsiMetadata: WsiMetadata;\r\n  visTemplates: VisualizationTemplates;\r\n  annotPresets: AnnotPresets;\r\n  jobConfig: JobConfig;\r\n\r\n  constructor(context: RationAI) {\r\n    this.context = context;\r\n    this.wsiMetadata = new WsiMetadata(this);\r\n    this.visTemplates = new VisualizationTemplates(this);\r\n    this.annotPresets = new AnnotPresets(this);\r\n    this.jobConfig = new JobConfig(this);\r\n  }\r\n\r\n  async get(itemId: string): Promise<GlobalItem> {\r\n    const globalItem: GlobalItem = await this.context.rawQuery(\r\n      `/global-storage/${itemId}`,\r\n    );\r\n    return globalItem;\r\n  }\r\n\r\n  async getValue(itemId: string): Promise<any> {\r\n    const item = await this.get(itemId);\r\n    if (item.type === 'string') {\r\n      try {\r\n        return JSON.parse(item.value);\r\n      } catch {\r\n        return item.value;\r\n      }\r\n    }\r\n    return item.value;\r\n  }\r\n\r\n  async query(query: GlobalStorageQuery): Promise<GlobalItem[]> {\r\n    const data: GlobalItemList = await this.context.rawQuery(\r\n      `/global-storage/query`,\r\n      {\r\n        method: 'PUT',\r\n        body: query,\r\n      },\r\n    );\r\n\r\n    return data.items;\r\n  }\r\n\r\n  async create(items: PostGlobalItems): Promise<GlobalItems> {\r\n    const createdItems: GlobalItems = await this.context.rawQuery(\r\n      `/global-storage`,\r\n      {\r\n        method: 'POST',\r\n        body: items,\r\n      },\r\n    );\r\n\r\n    return createdItems;\r\n  }\r\n\r\n  async createValue(\r\n    value: any,\r\n    name: string,\r\n    description?: string,\r\n    reference_id?: string,\r\n    reference_type?: GlobalItemReferenceType,\r\n    data_type?: string,\r\n  ): Promise<GlobalItem> {\r\n    value = JSON.stringify(value);\r\n\r\n    const newItem: PostGlobalItem = {\r\n      name: name,\r\n      description: description,\r\n      creator_id: this.context.userId,\r\n      creator_type: GlobalDataCreatorType.USER,\r\n      reference_id: reference_id,\r\n      reference_type: reference_type,\r\n      type: 'string',\r\n      value: value,\r\n      data_type: data_type,\r\n    };\r\n\r\n    return (await this.create(newItem)) as GlobalItem;\r\n  }\r\n\r\n  async update(itemId: string, item: PutGlobalItem): Promise<GlobalItem> {\r\n    const updatedItem: GlobalItem = await this.context.rawQuery(\r\n      `/global-storage/${itemId}`,\r\n      {\r\n        method: 'PUT',\r\n        body: item,\r\n      },\r\n    );\r\n    return updatedItem;\r\n  }\r\n\r\n  async updateValue(itemId: string, value: any): Promise<GlobalItem> {\r\n    const item: GlobalItem = await this.context.rawQuery(\r\n      `/global-storage/${itemId}`,\r\n    );\r\n    item.value = JSON.stringify(value);\r\n    const updatedItem: GlobalItem = await this.context.rawQuery(\r\n      `/global-storage/${itemId}`,\r\n      {\r\n        method: 'PUT',\r\n        body: item,\r\n      },\r\n    );\r\n    return updatedItem;\r\n  }\r\n\r\n  async delete(itemId: string): Promise<void> {\r\n    await this.context.rawQuery(`/global-storage/${itemId}`, {\r\n      method: 'DELETE',\r\n    });\r\n  }\r\n}\r\n","import { RawOptions } from '../../base';\r\nimport Root from '../root/root';\r\nimport GlobalStorage from './global-storage';\r\n\r\nexport default class RationAI {\r\n  static relativeApiPath = '/rationai';\r\n\r\n  context: Root;\r\n\r\n  //custom\r\n  globalStorage: GlobalStorage;\r\n\r\n  constructor(context: Root) {\r\n    this.context = context;\r\n    this.globalStorage = new GlobalStorage(this);\r\n  }\r\n\r\n  get userId(): string {\r\n    return this.context.userId;\r\n  }\r\n\r\n  async rawQuery(endpoint: string, options: RawOptions = {}): Promise<any> {\r\n    return this.context.rawQuery(\r\n      `${RationAI.relativeApiPath}${endpoint}`,\r\n      options,\r\n    );\r\n  }\r\n}\r\n","import {RootAPI} from \"../../root\";\r\n\r\nexport interface DefaultIntegrationOptions {\r\n    implementation?: string\r\n    _context?: RootAPI\r\n}\r\n\r\n/**\r\n * Integration class provides external information about users and their rights,\r\n * and relevant related external information\r\n */\r\nexport default class DefaultIntegration<T extends DefaultIntegrationOptions> {\r\n\r\n    protected context: RootAPI;\r\n    protected props: T;\r\n\r\n    constructor(props: T) {\r\n        if (props._context === undefined) {\r\n            throw Error(\"Private property not configured.\");\r\n        }\r\n        this.context = props._context;\r\n        this.props = props;\r\n    }\r\n\r\n    /**\r\n     * Maps local id parts to the specific name (see extensions)\r\n     * @param specKey see extension on case hierarchy keys / specs\r\n     * @param value\r\n     */\r\n    async translatePathSpec(specKey: string, value: string): Promise<string> {\r\n        return value;\r\n    }\r\n}\r\n\r\n","import {AuthIntegration} from \"./index\";\r\nimport {Constructor} from \"../extensions/utils\";\r\n\r\n/**\r\n * Integration manager / dealer\r\n */\r\nexport default class IntegrationManager {\r\n\r\n    private static items: Map<string, Constructor<AuthIntegration>> = new Map();\r\n\r\n    constructor() {\r\n       throw Error(\"Not instantiable.\");\r\n    }\r\n\r\n    static register(name: string, cls: Constructor<AuthIntegration>): Constructor<AuthIntegration> {\r\n        this.items.set(name, cls);\r\n        return cls;\r\n    }\r\n\r\n    static get(name: string): Constructor<AuthIntegration> | undefined {\r\n        return this.items.get(name);\r\n    }\r\n}\r\n","import DefaultIntegration, {DefaultIntegrationOptions} from \"./default\";\r\nimport LSAAIIntegration from \"./lsaai\";\r\nimport IntegrationManager from \"./integration-manager\";\r\n\r\nIntegrationManager.register(\"default\", DefaultIntegration);\r\nIntegrationManager.register(\"lsaai\", LSAAIIntegration);\r\nexport type AuthIntegration = DefaultIntegration<DefaultIntegrationOptions>;\r\nexport {IntegrationManager};\r\n","import DefaultIntegration, {DefaultIntegrationOptions} from \"./default\";\r\n\r\ninterface LSAAIIntegrationOptions extends DefaultIntegrationOptions {\r\n    userinfo: string\r\n}\r\n\r\ntype GroupName = {\r\n    name?: string,\r\n    slug?: string\r\n}\r\n\r\nexport default class LSAAIIntegration extends DefaultIntegration<LSAAIIntegrationOptions> {\r\n\r\n    private userData: any;\r\n\r\n    async translatePathSpec(specKey: string, value: string): Promise<string> {\r\n        if (!this.context.accessToken || !this.props.userinfo) {\r\n            return value;\r\n        }\r\n\r\n        if (!this.userData) {\r\n            await this.parseUserInstitutionsAndProjects();\r\n        }\r\n\r\n        if (specKey === \"id_part_1\") {\r\n            return this.userData.institutions[value] || value;\r\n        }\r\n        if (specKey === \"id_part_2\") {\r\n            return this.userData.projects[value] || value;\r\n        }\r\n        return value;\r\n    }\r\n\r\n    private parseGroupName (name: string): GroupName {\r\n        if (!name) return { name: name, slug: name }\r\n        name = decodeURIComponent(name).trim()\r\n        const match = name.match(/^(.+)\\.\\s*([^.\\s]{1,5})$/)\r\n        if (match && match.length === 3) {\r\n            return { name: match[1].trim(), slug: match[2].trim() }\r\n        }\r\n        if (name.length > 5) {\r\n            console.warn('Invalid group name', name)\r\n            return {}\r\n        }\r\n        return { name: name, slug: name }\r\n    }\r\n\r\n    private async parseUserInstitutionsAndProjects () {\r\n        const response = await fetch(this.props.userinfo, {\r\n            headers: {\r\n                'Authorization': `Bearer ${this.context.accessToken}`\r\n            }\r\n        });\r\n\r\n        const data = await response.text();\r\n        if (!response.ok) {\r\n            throw new Error(`Failed to fetch user info! ${response.statusText}. ${data}`);\r\n        }\r\n        const parsedData = JSON.parse(data);\r\n\r\n\r\n        const result = this.userData = {\r\n            projects: {\r\n                '': { label: 'NO PROJECT', value: '', rights: ['write'] }\r\n            },\r\n            institutions: {\r\n                '': { label: 'PUBLIC DATA', value: ''}\r\n\r\n            }\r\n        };\r\n\r\n        try {\r\n            const entitlements = parsedData.eduperson_entitlement\r\n            if (entitlements) {\r\n                // Parse AARCG069 for groups\r\n                for (const rule of entitlements) {\r\n                    // parse group (institution) hierarchy (todo hardcoded - env ?)\r\n                    const match = rule.match(/^.*?:group:ration_ai:([^#\\n]*)/)\r\n                    if (match && match.length > 1) {\r\n                        const groups = match[1]?.split(':')\r\n                        if (groups) {\r\n                            const length = groups.length,\r\n                                project = this.parseGroupName(groups[0]),\r\n                                organization = this.parseGroupName(length > 1 ? groups[1] : ''),\r\n                                rights = this.parseGroupName(length > 2 ? groups[2] : null)\r\n\r\n                            if (organization.slug && !result.institutions[organization.slug]) {\r\n                                result.institutions[organization.slug] = {\r\n                                    label: organization.name,\r\n                                    value: organization.slug\r\n                                }\r\n                            }\r\n                            if (project.slug && !result.projects[project.slug]) {\r\n                                result.projects[project.slug] = {\r\n                                    label: project.name,\r\n                                    value: project.slug,\r\n                                    rights: rights ? [rights] : []\r\n                                };\r\n                            } else if (rights && project.slug) {\r\n                                result.projects[project.slug].rights.push(rights.slug);\r\n                            }\r\n                        }\r\n                    } else {\r\n                        console.warn('Ignored entitlement', rule)\r\n                    }\r\n                }\r\n\r\n            } else {\r\n                console.warn('User info data does not contain access information! Is OAUTH scope set correctly?')\r\n            }\r\n        } catch (e) {\r\n            console.warn('Could not decide user authorization capabilities!', e)\r\n        }\r\n    }\r\n}\r\n","import { EmpationAPIOptions, RawAPI, RawOptions } from '../../base';\r\nimport { RootAPI } from '../../root';\r\nimport Scope from '../scope/scope';\r\nimport Apps from './apps';\r\nimport Cases from './cases';\r\nimport Examinations from './examinations';\r\nimport Slides from './slides';\r\nimport RationAI from '../rationai/rationai';\r\nimport { WorkbenchServiceApiV3CustomModelsExaminationsExamination } from './types/workbench-service-api-v-3-custom-models-examinations-examination';\r\n\r\n// Important to trigger all the index code!\r\nimport * as Integration from \"../integration\";\r\n\r\nexport default class Root extends RootAPI {\r\n  static apiPath = '/v3';\r\n\r\n  //interface\r\n  protected raw: RawAPI;\r\n  protected defaultScopeKey: string = '';\r\n  version: string;\r\n  rootURI: string;\r\n\r\n  integration: Integration.AuthIntegration;\r\n\r\n  scopes: Map<string, Scope>;\r\n\r\n  //custom\r\n  apps: Apps;\r\n  cases: Cases;\r\n  examinations: Examinations;\r\n  slides: Slides;\r\n  rationai: RationAI;\r\n\r\n  constructor(options: EmpationAPIOptions) {\r\n    super(options);\r\n    this.version = 'v3';\r\n    this.rootURI = this.options.apiUrl + Root.apiPath;\r\n    this.raw = new RawAPI(this.rootURI);\r\n\r\n\r\n    if (!options.integrationOptions) {\r\n      options.integrationOptions = {};\r\n    }\r\n\r\n    // Private this injection (instantiate has only one argument)\r\n    options.integrationOptions._context = this;\r\n\r\n    const provider = options.integrationOptions.implementation || \"default\";\r\n    const IntegrationClass = Integration.IntegrationManager.get(provider);\r\n    if (!IntegrationClass) {\r\n      throw new Error(`Could not instantiate integration provider ${provider} - is it a valid name?`);\r\n    }\r\n    this.integration = new IntegrationClass(options.integrationOptions);\r\n\r\n\r\n\r\n    this.apps = new Apps(this);\r\n    this.cases = new Cases(this);\r\n    this.examinations = new Examinations(this);\r\n    this.slides = new Slides(this);\r\n    this.rationai = new RationAI(this);\r\n\r\n    this.scopes = new Map<string, Scope>();\r\n  }\r\n\r\n  get defaultScope(): Scope | undefined {\r\n    return this.scopes.get(this.defaultScopeKey);\r\n  }\r\n\r\n  private async newScopeFrom(\r\n    examination: WorkbenchServiceApiV3CustomModelsExaminationsExamination,\r\n  ) {\r\n    const scope = new Scope(this);\r\n    await scope.from(examination);\r\n    this.scopes.set(examination.id, scope);\r\n    if (this.defaultScopeKey === '') this.defaultScopeKey = examination.id;\r\n    return scope;\r\n  }\r\n\r\n  private async newScopeUse(caseId: string, appId?: string) {\r\n    const scope = new Scope(this);\r\n    await scope.use(caseId, appId);\r\n    this.scopes.set(scope.activeExaminationId, scope);\r\n    if (this.defaultScopeKey === '')\r\n      this.defaultScopeKey = scope.activeExaminationId;\r\n    return scope;\r\n  }\r\n\r\n  async getScopeFrom(\r\n    examination: WorkbenchServiceApiV3CustomModelsExaminationsExamination,\r\n  ) {\r\n    return (\r\n      this.scopes.get(examination.id) || (await this.newScopeFrom(examination))\r\n    );\r\n  }\r\n\r\n  async getScopeUse(caseId: string, appId?: string) {\r\n    const matchingScopes = [...this.scopes.values()].filter(\r\n      (scp) =>\r\n        scp.activeCaseId === caseId &&\r\n        (appId ? scp.activeAppId === appId : true),\r\n    );\r\n    return matchingScopes.length > 0\r\n      ? matchingScopes[0]\r\n      : await this.newScopeUse(caseId, appId);\r\n  }\r\n\r\n  async rawQuery(endpoint: string, options: RawOptions = {}): Promise<any> {\r\n    await super.rawQuery(endpoint, options);\r\n    options = options || {};\r\n    options.headers = options.headers || {};\r\n    options.headers['User-Id'] = this.userId;\r\n    if (this.accessToken) {\r\n      options.headers['Authorization'] =\r\n        options.headers['Authorization'] || `Bearer ${this.rawToken}`;\r\n    }\r\n    return this.raw.http(endpoint, options);\r\n  }\r\n}\r\n"],"names":["webpackEmptyAsyncContext","req","Promise","resolve","then","e","Error","code","keys","id","module","exports","__webpack_module_cache__","__webpack_require__","moduleId","cachedModule","undefined","__webpack_modules__","d","definition","key","o","Object","defineProperty","enumerable","get","obj","prop","prototype","hasOwnProperty","call","r","Symbol","toStringTag","value","EventSource","events","addOnceHandler","eventName","handler","userData","times","priority","self","this","count","onceHandler","event","removeHandler","addHandler","isFunction","index","length","handlers","Array","isArray","i","push","numberOfHandlers","removeAllHandlers","eventType","getHandler","apply","source","args","eventSource","getAwaitingHandler","loop","result","type","raiseEvent","eventArgs","raiseEventAwaiting","awaitingHandler","String","class2type","toString","STATUS_CODES","HTTPError","message","extras","super","arguments","assign","name","suffix","replace","split","reduce","acc","c","charAt","toUpperCase","slice","httpErrorToName","statusCode","RawAPI","constructor","url","options","_parseQueryParams","params","k","v","URLSearchParams","_fetch","response","fetch","method","headers","body","responseType","status","statusText","error","ok","payload","http","endpoint","hasBody","startsWith","query","JSON","stringify","AbstractAPI","getCallerName","orig","prepareStackTrace","stack","requires","getJwtTokenExpiresTimeout","token","exp","Date","now","parseJwtToken","parse","atob","sleep","ms","setTimeout","Logger","console","warn","info","debug","RootContext","RootAPI","accessToken","_tokenExpires","_rawToken","workbenchApiUrl","apiUrl","apiRootPath","endsWith","anonymousUserId","_userId","cached","from","withEvent","reset","tokenTimeout","userId","sub","use","defaultScopeKey","scopes","forEach","scp","clear","rawToken","rawQuery","eventObject","newToken","ScopeContext","ScopeAPI","Apps","context","data","_defaultApp","list","apps","tissues","stains","job_modes","app","items","name_short","vendor_name","getNumber","variable","Number","matchStringOnSeparatorGroup","str","separator","groupIdx","matches","RegExp","exec","CaseExplorer","integration","customCases","caseHierarchy","caseTissues","caseStains","identifierSeparator","hierarchySpec","hierarchyNameOverrides","getCustomCases","map","caseObj","pathInHierarchy","getCaseHierarchyPath","pathFinished","prev","curr","val","getCaseValue","returnVal","getCase","caseId","find","cs","getCaseYear","getCaseMonth","getCaseDay","getCaseDescription","getCaseTissues","getCaseStains","isNaN","getCaseIdentifierPart","evaluateCaseValue","evalValue","caseValue","evaulateCaseYear","evaulateCaseMonth","evaulateCaseDay","evaluateCaseDescription","evaluateCaseTissues","evaluateCaseStains","evaulateCaseIdentifierPart","epochTime","created_at","getFullYear","getMonth","getDate","partIdx","parts","local_id","description","stringToMatch","tokenString","tokens","filter","Boolean","searchTermRegex","join","match","matchStringOnTokens","every","tissue","includes","stain","hierarchyLevel","keyIdx","cases","currentHierarchyPath","parent","levelId","levelName","lastLevel","groups","item","all","itemId","overrideName","translatePathSpec","child","hierarchy","search","filteredCases","localization","allTissues","entries","tisName","tisValue","locName","t","Map","values","allStains","stnName","stnValue","s","WsiExplorer","lastCaseId","slidesData","masksData","maskIdentifierSeparator","maskIdentifierValue","getAllSlides","slides","slide","masks","Cases","caseExplorer","wsiExplorer","Examinations","create","appId","case_id","app_id","skip","limit","examinationId","scope","Storage","getRaw","flush","content","set","valueRaw","dataRaw","erase","Annotations","withClasses","with_classes","createMany","deleteById","object","deleted","classes","cls","deleteClass","update","externalIds","addClass","addClassMany","getClass","queryClasses","Collections","collectionId","collection","queryItems","createItems","deleteItem","Jobs","getJobs","jobId","Pixelmaps","pixelmapId","post","pixelmaps","getTile","level","tileX","tileY","uploadTile","tile","deleteTile","bulkGetTile","startX","startY","endX","endY","bulkUploadTile","tiles","Scope","scopeContext","_defaultExaminationId","_tokenRefetchInterval","activeExaminationId","activeCaseId","activeAppId","raw","apiPath","storage","annotations","collections","jobs","findExamination","examinations","creators","item_count","examination","ex","state","default","scopeToken","access_token","scope_id","timeout","setInterval","clearInterval","GlobalDataCreatorType","GlobalItemReferenceType","TemplateType","Slides","slideInfo","slideId","slideThumbnail","maxWidth","maxHeight","format","image_format","slideLabel","loadTile","x","y","WsiMetadata","defaultSlideMetadata","visualization","defaultMaskMetadata","getWsiMetadataItem","wsiId","isSlide","metadata","references","p","data_type","createWsiMetadataItem","createValue","WSI","getSlideMetadata","updateSlideMetadata","metadataItem","updatedItem","getShadersConfig","shaders","shadersConfig","shaderTemp","visTemplates","getTemplate","Shader","shaderTemplate","dataReferences","dataRefs","getVisualizations","slideVis","paramsTemplate","Params","background","Background","template","dataRef","visualizations","vis","Visualization","visTemplate","getMaskMetadata","maskId","templatesGlobalItemDataType","VisualizationTemplates","fetchTemplateItem","data_types","tmpl","createTemplate","deleteTemplate","existingTmpl","delete","AnnotPresets","presetDataType","getPresetsItem","fresh","presetsItem","createPresetsItem","presets","getAnnotPresets","presetItem","lastModifiedAt","modified_at","mergePresets","primaryArr","secondaryArr","localVersion","newArr","secItem","some","primItem","createdAt","updateAnnotPresets","failOnParallelUpdate","remotePresetsItem","remotePresets","localPresets","successfulUpdate","retryAttempt","deleteAnnotPresets","JobConfig","configDataType","fetchJobConfigItem","getJobConfig","createJobConfig","JOB","deleteJobConfig","existingConfig","GlobalStorage","wsiMetadata","annotPresets","jobConfig","getValue","reference_id","reference_type","newItem","creator_id","creator_type","USER","updateValue","RationAI","globalStorage","relativeApiPath","DefaultIntegration","props","_context","specKey","IntegrationManager","register","userinfo","parseUserInstitutionsAndProjects","institutions","projects","parseGroupName","slug","decodeURIComponent","trim","text","parsedData","label","rights","entitlements","eduperson_entitlement","rule","project","organization","Root","version","rootURI","integrationOptions","provider","implementation","IntegrationClass","rationai","defaultScope","newScopeFrom","newScopeUse","getScopeFrom","getScopeUse","matchingScopes"],"sourceRoot":""} diff --git a/plugins/annotations/annotationsGUI.js b/plugins/annotations/annotationsGUI.js index 176f7602..ad74565a 100644 --- a/plugins/annotations/annotationsGUI.js +++ b/plugins/annotations/annotationsGUI.js @@ -3,8 +3,6 @@ class AnnotationsGUI extends XOpatPlugin { //todo test with multiple swap bgimages constructor(id) { super(id); - - this._server = this.getStaticMeta("server"); this._ioArgs = this.getStaticMeta("convertors") || {}; this._defaultFormat = this._ioArgs.format || "native"; this.registerAsEventSource(); @@ -86,6 +84,9 @@ class AnnotationsGUI extends XOpatPlugin { } this.enablePresetModify = this.getOptionOrConfiguration('enablePresetModify', 'enablePresetModify', true); + if (this.getOption("edgeCursorNavigate", true)) { + this.context.setCloseEdgeMouseNavigation(true); + } } setupActiveTissue(bgImageConfigObject) { @@ -104,8 +105,13 @@ class AnnotationsGUI extends XOpatPlugin { *****************************************************************************************************************/ - setDrawOutline(drawOutline) { - this.context.setAnnotationCommonVisualProperty('modeOutline', drawOutline); + setDrawOutline(enable) { + this.context.setAnnotationCommonVisualProperty('modeOutline', enable); + } + + setEdgeCursorNavigate(enable) { + this.setOption("edgeCursorNavigate", enable); + this.context.setCloseEdgeMouseNavigation(enable); } initHTML() { @@ -123,7 +129,7 @@ onclick="${this.THIS}._toggleEnabled(this)">visibility //
`, `
-
Border
+
Border
${UIComponents.Elements.checkBox({ label: this.t('outlineOnly'), classes: "pl-2", @@ -131,7 +137,13 @@ ${UIComponents.Elements.checkBox({ default: this.context.getAnnotationCommonVisualProperty('modeOutline')})}
-
Opacity
+
Opacity +${UIComponents.Elements.checkBox({ + label: 'Enable edge navigation', + classes: "pl-2", + onchange: `${this.THIS}.setEdgeCursorNavigate(!!this.checked)`, + default: this.getOption("edgeCursorNavigate", true)})} +
@@ -838,17 +850,27 @@ style="height: 22px; width: 60px;" onchange="${this.THIS}.context.freeFormTool.s }); } + /** + * Export annotations for one-time state save + * @param preferredFormat + * @param withObjects + * @param withPresets + * @return {Promise<*>} + */ + async getExportData(preferredFormat = null, withObjects=true, withPresets=true) { + this._ioArgs.format = preferredFormat || this._defaultFormat; + return this.context.export(this._ioArgs, withObjects, withPresets); + } + /** * Export annotations and download them */ exportToFile(withObjects=true, withPresets=true) { - const toFormat = this.exportOptions.format || this._defaultFormat; - this._ioArgs.format = toFormat; - + const toFormat = this.exportOptions.format; const name = APPLICATION_CONTEXT.referencedName(true) + "-" + UTILITIES.todayISOReversed() + "-" + (withPresets && withObjects ? "all" : (withObjects ? "annotations" : "presets")) - this.context.export(this._ioArgs, withObjects, withPresets).then(result => { + this.getExportData(toFormat, withObjects, withPresets).then(result => { UTILITIES.downloadAsFile(name + this.context.getFormatSuffix(toFormat), result); }).catch(e => { Dialogs.show("Could not export annotations in the selected format.", 5000, Dialogs.MSG_WARN); @@ -1164,13 +1186,18 @@ class="btn m-2">Set for left click
`: '
Stored on a server${error}
`; } - saveDefault() { - if (!this._server) { + async saveDefault() { + this.needsSave = false; + await this.raiseAwaitEvent('save-annotations', { + getData: this.getExportData.bind(this), + setNeedsDownload: (needsDownload) => { + this.needsSave = needsDownload; + } + }) + + if (this.needsSave) { this.exportToFile(); - return; } - Dialogs.show("Server-side storage is in the process of implementation. Please, save the data locally for now."); - //todo server upload!!! } } diff --git a/plugins/annotations/include.json b/plugins/annotations/include.json index 6babeec9..9db4825f 100644 --- a/plugins/annotations/include.json +++ b/plugins/annotations/include.json @@ -8,8 +8,6 @@ "includes" : ["annotationsGUI.js", "preview.js"], "modules": ["annotations", "human-readable-ids"], "permaLoad": false, - //Annotation server API endpoint to store annotations at - "server": null, //Available annotation object types "factories": ["polygon", "rect", "ellipse", "ruler", "text"], //Windowed history + annotation list diff --git a/plugins/empaia/empaia.js b/plugins/empaia/empaia.js index cd3bb6eb..0f1ac1fb 100644 --- a/plugins/empaia/empaia.js +++ b/plugins/empaia/empaia.js @@ -1,13 +1,62 @@ addPlugin('empaia', class extends XOpatPlugin { constructor(id) { super(id); + + //known cases will be the only ones supported this.cases = this.getOption('cases', {}); this.defaultAppId = this.getOption('appId', null); + this.api = EmpationAPI.V3.get(); this._currentCaseId = this._currentAppId = this.scopeAPI = null; this.api.__scope_def = [null, null]; + this.integrateWithPlugin('gui_annotations', async plugin => { + plugin.addHandler('save-annotations', async e => { + const Convertor = OSDAnnotations.Convertor.get("empaia"); + const convertor = new Convertor(plugin.context, {}); + const empaiaTiledImage = VIEWER.scalebar.getReferencedTiledImage(); + + const promises = []; + try { + const slideId = VIEWER.scalebar.getReferencedTiledImage()?.source?.getEmpaiaId(); + const annotations = await this.scopeAPI.annotations.query({creators: [this.scopeAPI.id], references: [slideId]}); + + const currentAnnotations = plugin.context.filter(_ => true); + for (let annotation of annotations.items) { + const match = currentAnnotations.findIndex(e => e.id === annotation.id); + if (match >= 0) { + currentAnnotations.splice(match, 1); + //todo try to replace if not equal + } else { + //todo what if sent for saving before hit save? remember annotations in processing + promises.push(this.scopeAPI.annotations.delete(annotation)); + } + } + + await Promise.all(promises); + + for (let annotation of currentAnnotations) { + //todo what if sent for saving before hit save? remember annotations in processing + console.warn("Found annotations that are not uploaded properly by the event routine!"); + if (!annotation.npp_created) { + //todo when testing some annotation did not have npp_created property + annotation.npp_created = 1e6; + } + const toUpload = convertor.encodeSingleObject(annotation, empaiaTiledImage.source); + const annot = await this.scopeAPI.annotations.create(toUpload); + annotation.id = annot.id; + } + + e.setNeedsDownload(false); + Dialogs.show("Saved."); + } catch (ex) { + this.alertSaveFailed(); + e.setNeedsDownload(true); + console.error(ex); + } + }); + }); this.integrateWithSingletonModule('annotations', async module => { EmpationAPI.integrateWithAnnotations(module); @@ -25,48 +74,67 @@ addPlugin('empaia', class extends XOpatPlugin { //todo some queue that performs updates one by one? - // module.addHandler('annotation-create', async ev => { - // const empaiaTiledImage = VIEWER.scalebar.getReferencedTiledImage(); - // const annotation = convertor.encodeSingleObject(ev.object, empaiaTiledImage.source); - // const annot = await this.scopeAPI.annotations.create(annotation); - // ev.object.id = annot.id; - // }); - // module.addHandler('annotation-delete', ev => { - // if (!ev.object.id) { - // console.warn("NO ID!") - // } - // this.scopeAPI.annotations.deleteById(ev.object.id); - // }); - // module.addHandler('annotation-replace', async ev => { - // if (ev.previous) { - // if (!ev.previous.id) { - // console.warn("NO ID!") - // } - // this.scopeAPI.annotations.deleteById(ev.previous.id); - // } - // if (ev.next) { - // ev.next.npp_created = ev.previous.npp_created; - // const empaiaTiledImage = VIEWER.scalebar.getReferencedTiledImage(); - // const annotation = convertor.encodeSingleObject(ev.next, empaiaTiledImage.source); - // const annot = await this.scopeAPI.annotations.create(annotation); - // ev.object.id = annot.id; - // } - // }); - // module.addHandler('annotation-edit', ev => { - // if (!ev.object.id) { - // console.warn("NO ID!") - // } - // const empaiaTiledImage = VIEWER.scalebar.getReferencedTiledImage(); - // const annotation = convertor.encodeSingleObject(ev.object, empaiaTiledImage.source); - // this.scopeAPI.annotations.update(annotation); - // }); + module.addHandler('annotation-create', async ev => { + try { + const empaiaTiledImage = VIEWER.scalebar.getReferencedTiledImage(); + const annotation = convertor.encodeSingleObject(ev.object, empaiaTiledImage.source); + const annot = await this.scopeAPI.annotations.create(annotation); + ev.object.id = annot.id; + } catch (e) { + console.error(e); + } + }); + module.addHandler('annotation-delete', ev => { + if (!ev.object.id) { + console.warn("NO ID!") + return; + } + + try { + this.scopeAPI.annotations.deleteById(ev.object.id); + } catch (e) { + console.error(e); + } + }); + module.addHandler('annotation-replace', async ev => { + try { + if (ev.previous) { + if (!ev.previous.id) { + console.warn("NO ID!") + return; + } + this.scopeAPI.annotations.deleteById(ev.previous.id); + } + if (ev.next) { + ev.next.npp_created = ev.previous.npp_created; + const empaiaTiledImage = VIEWER.scalebar.getReferencedTiledImage(); + const annotation = convertor.encodeSingleObject(ev.next, empaiaTiledImage.source); + const annot = await this.scopeAPI.annotations.create(annotation); + ev.next.id = annot.id; + } + } catch (e) { + console.error(e); + } + }); + module.addHandler('annotation-edit', ev => { + try { + if (!ev.object.id) { + console.warn("NO ID!"); + } + const empaiaTiledImage = VIEWER.scalebar.getReferencedTiledImage(); + const annotation = convertor.encodeSingleObject(ev.object, empaiaTiledImage.source); + this.scopeAPI.annotations.update(annotation); + } catch (e) { + console.error(e); + } + }); }); } async refreshScope(module) { const bgId = APPLICATION_CONTEXT.referencedId(); if (!this.setActiveScope(bgId)) { - //todo warn! + this.alertLoadFailed(); } else { this.scopeAPI = await this.api.newScopeUse(this._currentCaseId, this._currentAppId); @@ -79,25 +147,43 @@ addPlugin('empaia', class extends XOpatPlugin { async stateChanged(module) { if (!this.scopeAPI) { + this.alertLoadFailed(); return; } - //todo clear? - //todo loading screen? const slideId = VIEWER.scalebar.getReferencedTiledImage()?.source?.getEmpaiaId(); if (!slideId) { - //todo error message + this.alertLoadFailed(); return; } + + let dialogTimeout = setTimeout(() => { + dialogTimeout = null; + Dialogs.show("", 5000, Dialogs.MSG_WARN); + }, 1000); try { const annotations = await this.scopeAPI.annotations.query({creators: [this.scopeAPI.id], references: [slideId]}); + // todo clear annotations from possibly previous state? module.import(annotations, {format: "empaia"}, true); } catch (e) { - //todo err - throw e; + this.alertLoadFailed(); + console.error(e); + } + if (dialogTimeout) { + clearTimeout(dialogTimeout); + } else { + Dialogs.hide(); } } + alertLoadFailed() { + Dialogs.show(`Failed to load annotations from the server! Please, try to reload the page."`) + } + + alertSaveFailed() { + Dialogs.show(`Failed to load annotations from the server! Please, keep the local file with your annotations. It can be used to import them manually.`) + } + setActiveScope(wsiID) { if (!wsiID) { return false; diff --git a/server/php/init.php b/server/php/init.php index cc241208..7b3ac2b7 100644 --- a/server/php/init.php +++ b/server/php/init.php @@ -37,7 +37,10 @@ function safeReadPostValue($val) { if (!is_string($val)) return $val; try { - return json_decode($val); + $parsed = json_decode($val); + if ((bool)$val && $parsed != null) { + return $parsed; + } } catch (Exception $e) { return $val; } diff --git a/src/app.js b/src/app.js index 466e5416..1d2840bd 100644 --- a/src/app.js +++ b/src/app.js @@ -570,7 +570,7 @@ function initXopat(PLUGINS, MODULES, ENV, POST_DATA, PLUGINS_FOLDER, MODULES_FOL const magMicrons = microns || (micronsX + micronsY) / 2; // todo try read metadata about magnification and warn if we try to guess - const values = [70, 2, 15, 5, 7, 10, 0.5, 20, 0.25, 40]; + const values = [2.4, 2, 1.2, 4, 0.6, 10, 0.3, 20, 0.15, 40]; let index = 0, best = Infinity, mag; if (magMicrons) { while (index < values.length) { @@ -1241,8 +1241,14 @@ function initXopat(PLUGINS, MODULES, ENV, POST_DATA, PLUGINS_FOLDER, MODULES_FOL // Make sure the reference is really there POST_DATA.visualization = CONFIG; + // Clean up instance references before serialization + const plugins = {...PLUGINS}; + const modules = {...MODULES}; + for (let id in plugins) delete plugins[id].instance; + for (let id in modules) delete modules[id].instance; sessionStorage.setItem('__xopat_session__', JSON.stringify({ - PLUGINS, MODULES, ENV, POST_DATA, PLUGINS_FOLDER, MODULES_FOLDER, VERSION, I18NCONFIG + PLUGINS: plugins, MODULES: modules, + ENV, POST_DATA, PLUGINS_FOLDER, MODULES_FOLDER, VERSION, I18NCONFIG })); } catch (e) { diff --git a/src/external/scalebar.js b/src/external/scalebar.js index 247a5a7b..80fbf946 100644 --- a/src/external/scalebar.js +++ b/src/external/scalebar.js @@ -233,21 +233,25 @@ this.magnificationContainer.style.borderRadius = "7px"; - const steps = Math.round(Math.sqrt(this.magnification)) - 1; + let steps = 0; + let testMag = this.magnification; + while (testMag > 4) { + testMag = Math.round(testMag / 2); + steps++; + } + const minValue = 0; const sliderContainer = document.createElement("span"); const range = {max: [this.magnification], min: [1]}, values = [this.magnification]; - let mag = this.magnification, stepPerc = Math.round(89 / (steps-1)), stepPercIter = 100 - stepPerc; - while (mag > 5) { - mag = Math.round(mag / 2); + let mag = this.magnification, stepPerc = Math.round(100 / (steps+1)), stepPercIter = 100; + while (mag > 4) { + mag = Math.floor(mag / 2); + stepPercIter -= stepPerc; range[`${stepPercIter}%`] = [mag]; values.push(mag); - stepPercIter -= stepPerc; } - //few last step manually, make 2 more distant from home - range[`${stepPercIter+stepPerc*0.3}%`] = [2]; - values.push(2, 1); + values.push(1); values.reverse(); const updateZoom = (mag) => { diff --git a/src/loader.js b/src/loader.js index 5ae1e888..a20cb385 100644 --- a/src/loader.js +++ b/src/loader.js @@ -607,6 +607,7 @@ function initXOpatLoader(PLUGINS, MODULES, PLUGINS_FOLDER, MODULES_FOLDER, POST_ this.getHandler = events.getHandler.bind(events); this.numberOfHandlers = events.numberOfHandlers.bind(events); this.raiseEvent = events.raiseEvent.bind(events); + this.raiseAwaitEvent = VIEWER.tools.raiseAwaitEvent.bind(this, events); this.removeAllHandlers = events.removeAllHandlers.bind(events); this.removeHandler = events.removeHandler.bind(events); this.__errorBindingOnViewer = errorBindingOnViewer; @@ -673,6 +674,12 @@ function initXOpatLoader(PLUGINS, MODULES, PLUGINS_FOLDER, MODULES_FOLDER, POST_ * Note: noop if registerAsEventSource() not called. */ raiseEvent () {} + /** + * Trigger an event, optionally passing additional information. See OpenSeadragon.EventSource::raiseAwaitEvent. + * Awaits async handlers. + * Note: noop if registerAsEventSource() not called. + */ + raiseAwaitEvent() {} /** * Remove all event handlers for a given event type. See OpenSeadragon.EventSource::removeAllHandlers * Note: noop if registerAsEventSource() not called. diff --git a/src/parse-input.js b/src/parse-input.js index 1b1c127c..7cf5d3cf 100644 --- a/src/parse-input.js +++ b/src/parse-input.js @@ -159,9 +159,9 @@ function xOpatParseConfiguration(postData, i18n, supportsPost) { if (!session) { // Try to restore past state - const strData = window.localStorage.getItem("xoSessionCache"); - if (strData) { - const data = JSON.parse(strData); + let strData = window.localStorage.getItem("xoSessionCache"); + if (strData && strData !== "undefined") { + let data = JSON.parse(strData); // consider the session alive for at most 30 minutes const viz = data.visualization; if (viz && viz.__age && Date.now() - viz.__age < 1800e3) { @@ -169,6 +169,12 @@ function xOpatParseConfiguration(postData, i18n, supportsPost) { delete viz.__age; session = _parse(viz); session.__fromLocalStorage = true; + } else { + strData = window.sessionStorage.getItem("xoSessionCache"); + data = strData && strData !== "undefined" && JSON.parse(strData); + postData = data; + session = data.visualization && _parse(data.visualization); + session.__fromLocalStorage = true; } //window.localStorage.removeItem("xoSessionCache"); } @@ -177,7 +183,12 @@ function xOpatParseConfiguration(postData, i18n, supportsPost) { const data = postData || {}; session.__age = Date.now(); data.visualization = session; - window.localStorage.setItem("xoSessionCache", JSON.stringify(data)); + + const sessionData = JSON.stringify(data); + // Local Storage is meant for 'last session', available accross windows, session storage is to prevent + // losing context at any cost + window.localStorage.setItem("xoSessionCache", sessionData); + window.sessionStorage.setItem("xoSessionCache", sessionData); } // Todo this will make the viewer to not show any error - handled by the default screen... any better solution? From a10332ccb61cccc8ced59a95eabba7f487d1bc03 Mon Sep 17 00:00:00 2001 From: Aiosa <469130@mail.muni.cz> Date: Tue, 7 Jan 2025 11:08:18 +0100 Subject: [PATCH 2/6] chore: winrter school hot fix store preset IDs as classes too --- plugins/annotations/annotationsGUI.js | 3 +-- plugins/empaia/annotationConvertor.js | 26 ++++++++++++++++++++++---- plugins/empaia/empaia.js | 8 ++++++++ 3 files changed, 31 insertions(+), 6 deletions(-) diff --git a/plugins/annotations/annotationsGUI.js b/plugins/annotations/annotationsGUI.js index ad74565a..bb522a82 100644 --- a/plugins/annotations/annotationsGUI.js +++ b/plugins/annotations/annotationsGUI.js @@ -137,14 +137,13 @@ ${UIComponents.Elements.checkBox({ default: this.context.getAnnotationCommonVisualProperty('modeOutline')})}
-
Opacity +
Opacity
${UIComponents.Elements.checkBox({ label: 'Enable edge navigation', classes: "pl-2", onchange: `${this.THIS}.setEdgeCursorNavigate(!!this.checked)`, default: this.getOption("edgeCursorNavigate", true)})}
-
diff --git a/plugins/empaia/annotationConvertor.js b/plugins/empaia/annotationConvertor.js index 6e72b0ee..f3e621f1 100644 --- a/plugins/empaia/annotationConvertor.js +++ b/plugins/empaia/annotationConvertor.js @@ -101,7 +101,8 @@ EmpationAPI.integrateWithAnnotations = function (annotationsModule) { factoryID: "polygon", type: "polygon", npp_created: object.npp_created, - points: [object.head, object.tail] + points: [object.head, object.tail], + presetID: object.classes && object.classes.length && object.classes[0].value }), "rectangle": (object) => ({ id: object.id, @@ -112,6 +113,7 @@ EmpationAPI.integrateWithAnnotations = function (annotationsModule) { npp_created: object.npp_created, left: object.upper_left[0], top: object.upper_left[1], + presetID: object.classes && object.classes.length && object.classes[0].value }), "circle": (object) => ({ id: object.id, @@ -122,20 +124,23 @@ EmpationAPI.integrateWithAnnotations = function (annotationsModule) { npp_created: object.npp_created, left: object.center[0] - object.rx, top: object.center[1] - object.ry, + presetID: object.classes && object.classes.length && object.classes[0].value }), "polygon": (object) => ({ id: object.id, factoryID: "polygon", type: "polygon", npp_created: object.npp_created, - points: object.coordinates.map(p => ({x: p[0], y: p[1]})) + points: object.coordinates.map(p => ({x: p[0], y: p[1]})), + presetID: object.classes && object.classes.length && object.classes[0].value }), "line": (object) => ({ id: object.id, factoryID: "polygon", type: "polygon", npp_created: object.npp_created, - points: object.coordinates.map(p => ({x: p[0], y: p[1]})) + points: object.coordinates.map(p => ({x: p[0], y: p[1]})), + presetID: object.classes && object.classes.length && object.classes[0].value }), "point": (object) => ({ id: object.id, @@ -143,10 +148,23 @@ EmpationAPI.integrateWithAnnotations = function (annotationsModule) { type: "ellipse", npp_created: object.npp_created, left: object.coordinates[0], - top: object.coordinates[1] + top: object.coordinates[1], + presetID: object.classes && object.classes.length && object.classes[0].value }), } + getPreset(annotId, presetID) { + return { + creator_id: this.empaia.defaultScope.id, + creator_type: "scope", + reference_id: annotId, + reference_type: "annotation", + type: "class", + value: presetID + } + } + + _encodeAsEmpaiaObject(object, preset, tileSource, props) { //todo encode type and try to recover? ruler, text... return { diff --git a/plugins/empaia/empaia.js b/plugins/empaia/empaia.js index 0f1ac1fb..83dce728 100644 --- a/plugins/empaia/empaia.js +++ b/plugins/empaia/empaia.js @@ -46,6 +46,9 @@ addPlugin('empaia', class extends XOpatPlugin { const toUpload = convertor.encodeSingleObject(annotation, empaiaTiledImage.source); const annot = await this.scopeAPI.annotations.create(toUpload); annotation.id = annot.id; + + const preset = convertor.getPreset(annotation.id, annotation.presetID); + await this.scopeAPI.annotations.addClass(preset); } e.setNeedsDownload(false); @@ -80,6 +83,8 @@ addPlugin('empaia', class extends XOpatPlugin { const annotation = convertor.encodeSingleObject(ev.object, empaiaTiledImage.source); const annot = await this.scopeAPI.annotations.create(annotation); ev.object.id = annot.id; + const preset = convertor.getPreset(ev.object.id, ev.object.presetID); + await this.scopeAPI.annotations.addClass(preset); } catch (e) { console.error(e); } @@ -110,7 +115,10 @@ addPlugin('empaia', class extends XOpatPlugin { const empaiaTiledImage = VIEWER.scalebar.getReferencedTiledImage(); const annotation = convertor.encodeSingleObject(ev.next, empaiaTiledImage.source); const annot = await this.scopeAPI.annotations.create(annotation); + ev.next.id = annot.id; + const preset = convertor.getPreset(ev.next.id, ev.next.presetID); + await this.scopeAPI.annotations.addClass(preset); } } catch (e) { console.error(e); From 987cae9cf4e81c80bd142379715bd435a18f34fd Mon Sep 17 00:00:00 2001 From: Aiosa <469130@mail.muni.cz> Date: Tue, 7 Jan 2025 11:08:42 +0100 Subject: [PATCH 3/6] chore: bump version --- src/config.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/config.json b/src/config.json index 6083bef3..7b145605 100644 --- a/src/config.json +++ b/src/config.json @@ -5,7 +5,7 @@ { /**@lends xoEnv */ /* General xOpat Metadata */ "name": "xOpat", - "version": "2.1.0", + "version": "2.1.1-dev", /* Where xOpat redirects the user in case of error */ "gateway": "../", /* Active configuration in the "client" */ From 93515f9a9a4a0b84cfd8850a45a6a3c17b48e847 Mon Sep 17 00:00:00 2001 From: Aiosa <469130@mail.muni.cz> Date: Thu, 9 Jan 2025 11:58:46 +0100 Subject: [PATCH 4/6] fix: add missing export props, fix cache in getOption --- modules/annotations/annotations.js | 20 ++++---------------- modules/annotations/objects.js | 6 ++++-- plugins/annotations/include.json | 2 +- src/loader.js | 3 ++- 4 files changed, 11 insertions(+), 20 deletions(-) diff --git a/modules/annotations/annotations.js b/modules/annotations/annotations.js index 268e9846..4f0c7f76 100644 --- a/modules/annotations/annotations.js +++ b/modules/annotations/annotations.js @@ -898,7 +898,7 @@ window.OSDAnnotations = class extends XOpatModuleSingleton { * @param {boolean} defaultIfUnknown if false, empty string is returned in case no property was found * @return {string|*} annotation description */ - getAnnotationDescription(annotation, desiredKey="category", defaultIfUnknown=true) { + getAnnotationDescription(annotation, desiredKey="category", defaultIfUnknown=true, withCoordinates=true) { let preset = this.presets.get(annotation.presetID); if (preset) { for (let key in preset.meta) { @@ -906,11 +906,11 @@ window.OSDAnnotations = class extends XOpatModuleSingleton { let metaElement = preset.meta[key]; if (key === desiredKey) { return overridingValue || metaElement.value || - (defaultIfUnknown ? this.getDefaultAnnotationName(annotation) : ""); + (defaultIfUnknown ? this.getDefaultAnnotationName(annotation, withCoordinates) : ""); } } } - return defaultIfUnknown ? this.getDefaultAnnotationName(annotation) : ""; + return defaultIfUnknown ? this.getDefaultAnnotationName(annotation, withCoordinates) : ""; } /** @@ -1731,15 +1731,6 @@ window.OSDAnnotations = class extends XOpatModuleSingleton { } _loadObjects(input, clear, reviver, inheritSession) { - const originalToObject = fabric.Object.prototype.toObject; - const inclusionProps = this._exportedPropertiesGlobal(); - - //we ignore incoming props as we later reset the override - fabric.Object.prototype.toObject = function (_) { - return originalToObject.call(this, inclusionProps); - } - const resetToObjectCall = () => fabric.Object.prototype.toObject = originalToObject; - //from loadFromJSON implementation in fabricJS const _this = this.canvas, self = this; return new Promise((resolve, reject) => { @@ -1775,10 +1766,7 @@ window.OSDAnnotations = class extends XOpatModuleSingleton { return resolve(); }); }, reviver); - }).then(resetToObjectCall).catch(e => { - resetToObjectCall(); - throw e; - }); //todo rethrow? rewrite as async call with try finally + }); } _edgesMouseNavigation(e) { diff --git a/modules/annotations/objects.js b/modules/annotations/objects.js index 07a8fe7e..76e519a6 100644 --- a/modules/annotations/objects.js +++ b/modules/annotations/objects.js @@ -57,11 +57,13 @@ OSDAnnotations.AnnotationObjectFactory = class { "sessionID", "presetID", "layerID", - "id" + "id", + "author", + "created", ]; /** - * Properties copied with 'necessary' (+exports()) + * Properties copied with 'necessary' (+exports()), subset of copiedProperties * @type {string[]} */ static necessaryProperties = [ diff --git a/plugins/annotations/include.json b/plugins/annotations/include.json index 9db4825f..d8f45db3 100644 --- a/plugins/annotations/include.json +++ b/plugins/annotations/include.json @@ -5,7 +5,7 @@ "version": "1.0.0", "description": "A plugin for annotations creation, management and sharing.", "icon": null, - "includes" : ["annotationsGUI.js", "preview.js"], + "includes": ["annotationsGUI.js", "preview.js"], "modules": ["annotations", "human-readable-ids"], "permaLoad": false, //Available annotation object types diff --git a/src/loader.js b/src/loader.js index a20cb385..cfb43163 100644 --- a/src/loader.js +++ b/src/loader.js @@ -867,7 +867,7 @@ function initXOpatLoader(PLUGINS, MODULES, PLUGINS_FOLDER, MODULES_FOLDER, POST_ //todo allow APPLICATION_CONTEXT.getOption(...cache...) to disable cache globally //options are stored only for plugins, so we store them at the lowest level - let value = cache ? localStorage.getItem(`${this.id}.${key}`) : null; + let value = cache ? this.cache.get(key, null) : null; if (value === null) { // read default value from static context if exists if (defaultValue === undefined && key !== "instance") { @@ -885,6 +885,7 @@ function initXOpatLoader(PLUGINS, MODULES, PLUGINS_FOLDER, MODULES_FOLDER, POST_ /** * Ability to cache a value locally into the browser, * the value can be retrieved using this.getOption(...) + * todo rename to setCacheOption * @param key * @param value */ From c3f0967137110a767019e6f46ce19ed6129d1207 Mon Sep 17 00:00:00 2001 From: Aiosa <469130@mail.muni.cz> Date: Thu, 9 Jan 2025 12:06:49 +0100 Subject: [PATCH 5/6] fix: rely on session storage to allow multiple side tabs refresh --- src/parse-input.js | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/src/parse-input.js b/src/parse-input.js index 7cf5d3cf..4cf782bf 100644 --- a/src/parse-input.js +++ b/src/parse-input.js @@ -161,7 +161,7 @@ function xOpatParseConfiguration(postData, i18n, supportsPost) { // Try to restore past state let strData = window.localStorage.getItem("xoSessionCache"); if (strData && strData !== "undefined") { - let data = JSON.parse(strData); + const data = JSON.parse(strData); // consider the session alive for at most 30 minutes const viz = data.visualization; if (viz && viz.__age && Date.now() - viz.__age < 1800e3) { @@ -169,14 +169,14 @@ function xOpatParseConfiguration(postData, i18n, supportsPost) { delete viz.__age; session = _parse(viz); session.__fromLocalStorage = true; - } else { - strData = window.sessionStorage.getItem("xoSessionCache"); - data = strData && strData !== "undefined" && JSON.parse(strData); - postData = data; - session = data.visualization && _parse(data.visualization); - session.__fromLocalStorage = true; } - //window.localStorage.removeItem("xoSessionCache"); + window.localStorage.removeItem("xoSessionCache"); + } else { + strData = window.sessionStorage.getItem("xoSessionCache"); + const data = strData && strData !== "undefined" && JSON.parse(strData); + postData = data; + session = data.visualization && _parse(data.visualization); + session.__fromLocalStorage = true; } } else if (!session.error) { // Save current state (including post) in case we loose it and need to restore it (e.g. auth redirect) From a7cfe8894af8b59c56d80286b3201414b31d2e07 Mon Sep 17 00:00:00 2001 From: Aiosa <469130@mail.muni.cz> Date: Tue, 21 Jan 2025 16:04:22 +0100 Subject: [PATCH 6/6] fix: empaia standalone protocol correct query --- docs/web/xopat_deployment.md | 7 ++++--- modules/empaia-wsi-tile-source/tile-source.js | 8 +++++--- 2 files changed, 9 insertions(+), 6 deletions(-) diff --git a/docs/web/xopat_deployment.md b/docs/web/xopat_deployment.md index 8fd682e9..63db4e7a 100644 --- a/docs/web/xopat_deployment.md +++ b/docs/web/xopat_deployment.md @@ -25,10 +25,11 @@ To deploy xOpat, we need to configure it statically, "path": "/", // The default image server used. Configures an OpenSeadragon protocol using here URL of the service "image_group_server": "http://localhost:8080", - "image_group_protocol": "`${path}/v3/batch/info?slides=${data}`", - "image_group_preview": "`${path}/v3/batch/thumbnail/max_size/250/250?slides=${data}`", + "image_group_protocol": "`${path}/v3/slides/info?slide_id=${data}`", + "image_group_preview": "`${path}/v3/slides/thumbnail/max_size/250/250?slide_id=${data}`", "data_group_server": "http://localhost:8080", - "data_group_protocol": "`${path}/v3/batch/info?slides=${data.join(\",\")}`", + // This endpoint needs to ask for array of data items (get me tile level 5 x3 y0 for this slide list) + "data_group_protocol": "`${path}/v3/files/info?paths=${data.join(\",\")}`", "headers": {}, "js_cookie_expire": 365, "js_cookie_path": "/", diff --git a/modules/empaia-wsi-tile-source/tile-source.js b/modules/empaia-wsi-tile-source/tile-source.js index 62872495..5683c0fd 100644 --- a/modules/empaia-wsi-tile-source/tile-source.js +++ b/modules/empaia-wsi-tile-source/tile-source.js @@ -29,7 +29,7 @@ OpenSeadragon.EmpaiaStandaloneV3TileSource = class extends OpenSeadragon.TileSou let match = url.match(/^(\/?[^\/].*\/v3\/)(files|batch)\/info/i); if (match) { data = data || [{}]; - data[0].tilesUrl = match[1] + "batch"; + data[0].tilesUrl = match[1] + match[2]; return true; } } else if (url && typeof data === "object") { @@ -211,10 +211,12 @@ OpenSeadragon.EmpaiaStandaloneV3TileSource = class extends OpenSeadragon.TileSou if (this.multifetch) { //endpoint files/tile/level/[L]/tile/[X]/[Y]/?paths=path,list,separated,by,commas - return `${tiles}/tile/level/${level}/tile/${x}/${y}?slides=${this.fileId}` + const query_name = tiles.endsWith("batch") ? "slides" : "paths"; + return `${tiles}/tile/level/${level}/tile/${x}/${y}?${query_name}=${this.fileId}` } //endpoint slides/[SLIDE]/tile/level/[L]/tile/[X]/[Y]/ - return `${tiles}/tile/level/${level}/tile/${x}/${y}?slide=${this.fileId}` + const query_name = tiles.endsWith("batch") ? "slides" : "slide_id"; + return `${tiles}/tile/level/${level}/tile/${x}/${y}?${query_name}=${this.fileId}` } _setDownloadHandler(isMultiplex) {