@datagrok/peptides 1.17.21 → 1.17.22
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +4 -0
- package/dist/111.js +1 -1
- package/dist/111.js.map +1 -0
- package/dist/216.js +1 -1
- package/dist/216.js.map +1 -0
- package/dist/501.js +1 -1
- package/dist/501.js.map +1 -0
- package/dist/603.js +1 -1
- package/dist/603.js.map +1 -0
- package/dist/682.js +1 -1
- package/dist/682.js.map +1 -0
- package/dist/705.js +1 -1
- package/dist/705.js.map +1 -0
- package/dist/778.js +1 -1
- package/dist/778.js.map +1 -0
- package/dist/795.js +1 -1
- package/dist/795.js.map +1 -0
- package/dist/950.js +1 -1
- package/dist/950.js.map +1 -0
- package/dist/package-test.js +2 -2
- package/dist/package-test.js.map +1 -0
- package/dist/package.js +2 -2
- package/dist/package.js.map +1 -0
- package/package.json +1 -1
- package/src/viewers/cluster-max-activity-viewer.ts +3 -1
- package/src/viewers/logo-summary.ts +4 -1
- package/src/widgets/mutation-cliffs.ts +3 -0
- package/tsconfig.json +3 -3
- package/webpack.config.js +1 -1
package/dist/795.js
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
var peptides;(()=>{"use strict";var e,n={m:{},u:e=>e+".js"};n.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),n.o=(e,n)=>Object.prototype.hasOwnProperty.call(e,n),(()=>{var e;n.g.importScripts&&(e=n.g.location+"");var t=n.g.document;if(!e&&t&&(t.currentScript&&(e=t.currentScript.src),!e)){var r=t.getElementsByTagName("script");if(r.length)for(var a=r.length-1;a>-1&&(!e||!/^http(s?):/.test(e));)e=r[a--].src}if(!e)throw new Error("Automatic publicPath is not supported in this browser");e=e.replace(/#.*$/,"").replace(/\?.*$/,"").replace(/\/[^\/]+$/,"/"),n.p=e})(),n.b=self.location+"",function(e){e.EUCLIDEAN="EUCLIDEAN",e.MANHATTAN="MANHATTAN"}(e||(e={}));const t={[e.EUCLIDEAN]:function(e){return`\n var sum = 0.0;\n for (var i = 0u; i < ${e}; i = i + 1u) {\n sum = sum + distances[i] * distances[i] * computeInfo.weights[i] * computeInfo.weights[i];\n }\n return sqrt(sum);\n `},[e.MANHATTAN]:function(e){return`\n var sum = 0.0;\n for (var i = 0u; i < ${e}; i = i + 1u) {\n sum = sum + abs(distances[i]) * computeInfo.weights[i];\n }\n return sum;\n `}};var r;!function(e){e.HAMMING="Hamming",e.EUCLIDEAN="Euclidean",e.MANHATTAN="Manhattan",e.TANIMOTO="Tanimoto",e.LEVENSTEIN="Levenshtein",e.NEEDLEMAN_WUNSCH="Needlemann-Wunsch",e.MONOMER_CHEMICAL_DISTANCE="Monomer chemical distance",e.SOKAL="Sokal",e.COSINE="Cosine",e.ASYMMETRIC="Asymmetric",e.Difference="Difference",e.OneHot="One-Hot"}(r||(r={}));const a={[r.HAMMING]:function(e,n){return`\n let aLength: u32 = computeInfo.entrySizes[${n}][aIndex];\n let bLength: u32 = computeInfo.entrySizes[${n}][bIndex];\n let maxLength: u32 = max(aLength, bLength);\n let minLength: u32 = min(aLength, bLength);\n let sizeDiff: u32 = maxLength - minLength;\n \n let maxIntDistance = ceil(maxDistance * f32(maxLength)) - f32(sizeDiff);\n\n var diff: f32 = 0.0;\n for (var i = 0u; i < ${e}; i = i + 1u) {\n diff = diff + f32(a[i] != b[i]);\n if (diff > maxIntDistance) {\n return 1.0;\n }\n }\n diff += f32(sizeDiff);\n return diff / ${e};\n `},[r.EUCLIDEAN]:function(e,n){return`\n var dist: f32 = 0.0;\n for (var i = 0u; i < ${e}; i = i + 1u) {\n dist = dist + f32(a[i] - b[i]) * f32(a[i] - b[i]);\n }\n return sqrt(dist);\n `},[r.MANHATTAN]:function(e,n){return`\n var dist: f32 = 0.0;\n for (var i = 0u; i < ${e}; i = i + 1u) {\n dist = dist + abs(f32(a[i] - b[i]));\n }\n return dist;\n `},[r.TANIMOTO]:function(e,n){return`\n var onBitsa: u32 = 0u;\n var onBitsb: u32 = 0u;\n for (var i = 0u; i < ${e}u; i = i + 1u) {\n onBitsa = onBitsa + countOneBits(a[i]);\n onBitsb = onBitsb + countOneBits(b[i]);\n }\n\n if (onBitsa == 0u && onBitsb == 0u) {\n return 0.0;\n }\n\n let totalOnBits = onBitsa + onBitsb;\n var commonBits: u32 = 0u;\n for (var i = 0u; i < ${e}u; i = i + 1u) {\n commonBits = commonBits + countOneBits(a[i] & b[i]);\n }\n\n return 1.0 - f32(commonBits) / f32(totalOnBits - commonBits);\n `},[r.LEVENSTEIN]:function(e,n){return`\n let aLength: u32 = computeInfo.entrySizes[${n}][aIndex];\n let bLength: u32 = computeInfo.entrySizes[${n}][bIndex];\n let maxLength: u32 = max(aLength, bLength);\n let minLength: u32 = min(aLength, bLength);\n\n let maxIntDistance = ceil(maxDistance * f32(maxLength));\n\n // we will store two arrays as matrix and swap the working indices per pass.\n // this way we can reduce memory usage per computation to just O(aLength)\n // the grid will have aLength + 1 columns and bLength + 1 rows\n // this will be guaranteed by iteration, but the array sizes must be known at compile time, so we will use a fixed size of maxArraySize\n var dynamicPassMat: array<array<f32, ${e+1}u>, 2>; // initialize to 0\n \n var prevIndex: u32 = 0;\n var curIndex: u32 = 1; // we will swap these indices per pass\n\n // initialize the first row\n for (var i = 0u; i <= aLength; i = i + 1u) {\n dynamicPassMat[prevIndex][i] = f32(i);\n }\n\n // iterate over the rows\n for (var i = 1u; i <= bLength; i = i + 1u) {\n dynamicPassMat[curIndex][0] = f32(i);\n var minEntry: f32 = f32(maxLength);\n let prevRow = &dynamicPassMat[prevIndex];\n let curRow = &dynamicPassMat[curIndex];\n let bMon = u32(b[i - 1]);\n for (var j = 1u; j <= aLength; j = j + 1u) {\n var cost: f32 = f32(a[j - 1] != bMon);\n var res: f32 = min(\n min(\n (*prevRow)[j] + 1.0, // deletion\n (*curRow)[j - 1] + 1.0, // insertion\n ),\n (*prevRow)[j - 1] + cost // substitution\n );\n (*curRow)[j] = res;\n if (res < minEntry) {\n minEntry = res;\n }\n }\n // swap the indices\n let temp: u32 = prevIndex;\n prevIndex = curIndex;\n curIndex = temp;\n if (minEntry > maxIntDistance) {\n return 1.0;\n }\n }\n\n return dynamicPassMat[prevIndex][aLength] / f32(maxLength);\n `},[r.NEEDLEMAN_WUNSCH]:function(e,n){return`\n let aLength: u32 = computeInfo.entrySizes[${n}][aIndex];\n let bLength: u32 = computeInfo.entrySizes[${n}][bIndex];\n let maxLength: u32 = max(aLength, bLength);\n let minLength: u32 = min(aLength, bLength);\n \n let maxIntDistance = ceil(maxDistance * f32(maxLength));\n // we will store two arrays as matrix and swap the working indices per pass.\n // this way we can reduce memory usage per computation to just O(aLength)\n // the grid will have aLength + 1 columns and bLength + 1 rows\n // this will be guaranteed by iteration, but the array sizes must be known at compile time, so we will use a fixed size of maxArraySize\n var dynamicPassMat: array<array<f32, ${e+1}u>, 2>; // initialize to 0\n \n // we need to keep track of which operation led to the current cell\n // i.e. whether we came from the left, top or diagonal to assign gap open/gap extend penalty\n var verticalGaps: array<u32, ${e+1}u>;\n var horizontalGaps: array<u32, ${e+1}u>;\n\n let gapOpenPenalty: f32 = suppInfo.gapOpenPenalty${n};\n let gapExtensionPenalty: f32 = suppInfo.gapExtensionPenalty${n};\n var prevIndex: u32 = 0;\n var curIndex: u32 = 1; // we will swap these indices per pass\n // initialize the first row\n for (var i = 0u; i <= aLength; i = i + 1u) {\n dynamicPassMat[prevIndex][i] = gapOpenPenalty + f32(i - 1) * gapExtensionPenalty;\n dynamicPassMat[curIndex][i] = 0.0;\n }\n dynamicPassMat[0][0] = 0.0;\n\n let simMatrix = &suppInfo.similarityMatrix${n}; // using pointers make things faster\n // iterate over the rows\n for (var i = 1u; i <= bLength; i = i + 1u) {\n let prevRow = &dynamicPassMat[prevIndex];\n let curRow = &dynamicPassMat[curIndex];\n (*curRow)[0] = gapOpenPenalty + f32(i - 1) * gapExtensionPenalty;\n var minEntry: f32 = f32(maxLength);\n let monB = u32(b[i - 1]);\n for (var j = 1u; j <= aLength; j = j + 1u) {\n let monA = u32(a[j - 1]);\n \n let cost: f32 = (*prevRow)[j - 1] + 1f - (*simMatrix)[monA][monB];\n var top = (*prevRow)[j]; // deletion\n if (verticalGaps[j] > 0) {\n top = top + gapExtensionPenalty;\n } else {\n top = top + gapOpenPenalty;\n }\n var left = (*curRow)[j - 1]; // insertion\n if (horizontalGaps[j - 1] > 0) {\n left = left + gapExtensionPenalty;\n } else {\n left = left + gapOpenPenalty;\n }\n var res: f32 = min(\n min(\n top, // deletion\n left, // insertion\n ),\n cost // substitution\n );\n (*curRow)[j] = res;\n if (res < minEntry) {\n minEntry = res;\n }\n // update the horizontal and vertical gaps\n if (res == cost) {\n verticalGaps[j] = 0;\n horizontalGaps[j] = 0;\n } else if (res == left) {\n verticalGaps[j] = 0;\n horizontalGaps[j] = 1;\n } else {\n verticalGaps[j] = 1;\n horizontalGaps[j] = 0;\n }\n }\n // swap the indices\n let temp: u32 = prevIndex;\n prevIndex = curIndex;\n curIndex = temp;\n if (minEntry > maxIntDistance) {\n return 1.0;\n }\n }\n return dynamicPassMat[prevIndex][aLength] / f32(maxLength);\n \n `},[r.MONOMER_CHEMICAL_DISTANCE]:function(e,n){return`\n let aLength: u32 = computeInfo.entrySizes[${n}][aIndex];\n let bLength: u32 = computeInfo.entrySizes[${n}][bIndex];\n let maxLength: u32 = max(aLength, bLength);\n let minLength: u32 = min(aLength, bLength);\n let sizeDiff: u32 = maxLength - minLength;\n \n let maxIntDistance = ceil(maxDistance * f32(maxLength)) - f32(sizeDiff);\n\n let simMatrix = &(suppInfo.similarityMatrix${n}); // using pointers make things faster\n var diff: f32 = 0.0;\n for (var i = 0u; i < ${e}; i = i + 1u) {\n diff = diff + 1.0 - (*simMatrix)[u32(a[i])][u32(b[i])];\n if (diff > maxIntDistance) {\n return 1.0;\n }\n }\n diff += f32(sizeDiff);\n return diff / ${e};\n `},[r.SOKAL]:function(e,n){return`\n var onBitsa: u32 = 0u;\n var onBitsb: u32 = 0u;\n for (var i = 0u; i < ${e}u; i = i + 1u) {\n onBitsa = onBitsa + countOneBits(a[i]);\n onBitsb = onBitsb + countOneBits(b[i]);\n }\n let total = onBitsa + onBitsb;\n if (total == 0u) {\n return 1.0;\n }\n var commonBits: u32 = 0u;\n for (var i = 0u; i < ${e}u; i = i + 1u) {\n commonBits = commonBits + countOneBits(a[i] & b[i]);\n }\n return 1.0 - f32(commonBits) / f32(total * 2 - commonBits * 3);\n `},[r.COSINE]:function(e,n){return`\n var onBitsa: u32 = 0u;\n var onBitsb: u32 = 0u;\n for (var i = 0u; i < ${e}u; i = i + 1u) {\n onBitsa = onBitsa + countOneBits(a[i]);\n onBitsb = onBitsb + countOneBits(b[i]);\n }\n let total = onBitsa * onBitsb; // p.s. here total is taken by multiplying\n if (total == 0u) {\n return 1.0;\n }\n var commonBits: u32 = 0u;\n for (var i = 0u; i < ${e}u; i = i + 1u) {\n commonBits = commonBits + countOneBits(a[i] & b[i]);\n }\n return 1.0 - f32(commonBits) / sqrt(f32(total));\n `},[r.ASYMMETRIC]:function(e,n){return`\n var onBitsa: u32 = 0u;\n var onBitsb: u32 = 0u;\n for (var i = 0u; i < ${e}u; i = i + 1u) {\n onBitsa = onBitsa + countOneBits(a[i]);\n onBitsb = onBitsb + countOneBits(b[i]);\n }\n let min = min(onBitsa, onBitsb);\n if (min == 0u) {\n return 1.0;\n }\n var commonBits: u32 = 0u;\n for (var i = 0u; i < ${e}u; i = i + 1u) {\n commonBits = commonBits + countOneBits(a[i] & b[i]);\n }\n return 1.0 - f32(commonBits) / f32(min);\n `},[r.Difference]:function(e,n){return`\n let range = suppInfo.range${n};\n return f32(abs(f32(a[0]) - f32(b[0])) / range);\n `},[r.OneHot]:function(e,n){return`\n let aLength: u32 = computeInfo.entrySizes[${n}][aIndex];\n let bLength: u32 = computeInfo.entrySizes[${n}][bIndex];\n if (aLength != bLength) {\n return 1.0;\n }\n for (var i = 0u; i < aLength; i = i + 1u) {\n if(a[i] != b[i]) {\n return 1.0;\n }\n }\n return 0.0;\n `}},i={[r.HAMMING]:e=>Math.ceil(e/30),[r.EUCLIDEAN]:e=>Math.ceil(e/30),[r.MANHATTAN]:e=>Math.ceil(e/30),[r.TANIMOTO]:e=>Math.ceil(e/60),[r.SOKAL]:e=>Math.ceil(e/60),[r.COSINE]:e=>Math.ceil(e/60),[r.ASYMMETRIC]:e=>Math.ceil(e/60),[r.LEVENSTEIN]:e=>Math.ceil(e*e/60),[r.NEEDLEMAN_WUNSCH]:e=>Math.ceil(e*e/60),[r.MONOMER_CHEMICAL_DISTANCE]:e=>Math.ceil(e/25),[r.Difference]:e=>1,[r.OneHot]:e=>Math.ceil(e/40)},o={STRING:new Set([r.HAMMING,r.LEVENSTEIN,r.NEEDLEMAN_WUNSCH,r.MONOMER_CHEMICAL_DISTANCE,r.OneHot]),UINT32ARRAY:new Set([r.HAMMING,r.EUCLIDEAN,r.MANHATTAN,r.MONOMER_CHEMICAL_DISTANCE,r.LEVENSTEIN,r.NEEDLEMAN_WUNSCH,r.TANIMOTO,r.COSINE,r.SOKAL,r.ASYMMETRIC,r.OneHot,r.Difference]),INT32ARRAY:new Set([r.EUCLIDEAN,r.MANHATTAN,r.OneHot,r.Difference]),FLOAT32ARRAY:new Set([r.EUCLIDEAN,r.MANHATTAN,r.Difference]),NUMBER:new Set([r.EUCLIDEAN,r.MANHATTAN,r.Difference]),BITARRAY:new Set([r.TANIMOTO,r.COSINE,r.SOKAL,r.ASYMMETRIC])};var s=function(e,n,t,r){return new(t||(t=Promise))((function(a,i){function o(e){try{l(r.next(e))}catch(e){i(e)}}function s(e){try{l(r.throw(e))}catch(e){i(e)}}function l(e){var n;e.done?a(e.value):(n=e.value,n instanceof t?n:new t((function(e){e(n)}))).then(o,s)}l((r=r.apply(e,n||[])).next())}))};let l=null,u=null;function f(){return s(this,void 0,void 0,(function*(){if(!l&&(l=yield navigator.gpu.requestAdapter({powerPreference:"high-performance"}),null==l))return null;let e=!1;if(u&&(u.lost.then((()=>{e=!0})),yield new Promise((e=>setTimeout(e,10)))),!u||e){const e=1e9,n=l.limits,t=n.maxBufferSize,r=n.maxStorageBufferBindingSize;try{return u=yield l.requestDevice({requiredLimits:{maxBufferSize:Math.min(t,e),maxStorageBufferBindingSize:Math.min(r,e)}}),u}catch(e){return console.error("Failed to create device with required limits",e),u=yield l.requestDevice(),u}}return u}))}var c,d;function h(n,t=.8,a,s,l,u){return c=this,d=void 0,p=function*(){const c=yield f();if(!c)return null;const d=Object.values(r);if(a.some((e=>!d.includes(e))))throw new Error("Invalid distance metrics provided: "+a.join(", "));if(!Object.values(e).includes(s))throw new Error("Invalid aggregation function provided: "+s);const h=1-t;if(u.length!==n.length||u.length!==a.length||u.length!==l.length)throw new Error("Options, weigths and distance functions must be provided for each column");if(n.some((e=>e.length!==n[0].length)))throw new Error("All entry lists must be the same length");const p=n.length,g=n[0].length,y=n.map(((e,n)=>function(e,n=r.HAMMING,t,a={gapOpenPenalty:1,gapExtensionPenalty:.6}){var s,l;let u=null;const f=e.some((e=>"string"==typeof e))?(u="STRING",e.map((e=>new Uint32Array(e.split("").map((e=>e.charCodeAt(0))))))):e.some((e=>"number"==typeof e))?(u="NUMBER",e.map((e=>new Float32Array([e])))):"object"==typeof e[0]&&e.some((e=>"_data"in e&&"_length"in e))?(u="BITARRAY",e.map((e=>e._data))):e.some((e=>e instanceof Float32Array))?(u="FLOAT32ARRAY",e):e.some((e=>e instanceof Uint32Array))?(u="UINT32ARRAY",e):e.some((e=>e instanceof Int32Array))?(u="INT32ARRAY",e):void 0;if(!f||!u)throw new Error("Invalid entry type, could not determine entry type from input list");const c=f[0]instanceof Int32Array?"INT32ARRAY":f[0]instanceof Float32Array?"FLOAT32ARRAY":"UINT32ARRAY",d=new Uint32Array(f.map((e=>e.length)));if(!o[u]||!o[u].has(n))throw new Error(`Distance metric '${n}' not supported for entry type '${u}'`);const h=d.reduce(((e,n)=>Math.max(e,n)),0),m=i[n](h),p="INT32ARRAY"===c?Int32Array:"FLOAT32ARRAY"===c?Float32Array:Uint32Array,g=new p(f.length*h);f.forEach(((e,n)=>{g.set(e,n*h)}));let y="",w=0,A="FLOAT32ARRAY",b=null;if(n===r.NEEDLEMAN_WUNSCH||n===r.MONOMER_CHEMICAL_DISTANCE){let e=a.scoringMatrix&&a.alphabetIndexes?Object.keys(a.alphabetIndexes).reduce(((e,n)=>Math.max(e,n.charCodeAt(0))),0):-1;if(!a.alphabetIndexes||!a.scoringMatrix){for(let n=0;n<g.length;n++)g[n]>e&&(e=g[n]);a.scoringMatrix=new Array(e+1).fill(null).map((()=>new Array(e+1).fill(0))),a.alphabetIndexes={};for(let e=0;e<a.scoringMatrix.length;e++)a.scoringMatrix[e][e]=1,a.alphabetIndexes[String.fromCharCode(e)]=e}const n=(e+1)*(e+1),r=new Array(e+1).fill(null).map((()=>new Float32Array(e+1)));for(let n=0;n<e+1;n++)r[n][n]=1;const i=a.alphabetIndexes;for(const e of Object.keys(i))for(const n of Object.keys(i))e!==n&&(r[e.charCodeAt(0)][n.charCodeAt(0)]=a.scoringMatrix[i[e]][i[n]]);w=2+n,A="FLOAT32ARRAY",b=new Float32Array(w),b[0]=null!==(s=a.gapOpenPenalty)&&void 0!==s?s:1,b[1]=null!==(l=a.gapExtensionPenalty)&&void 0!==l?l:.6;let o=2;for(let e=0;e<r.length;e++)b.set(r[e],o),o+=r[e].length;y=`\n gapOpenPenalty${t}: f32,\n gapExtensionPenalty${t}: f32,\n similarityMatrix${t}: array<array<f32, ${e+1}>, ${e+1}>`}else if(n===r.Difference){if(!a.range||"number"!=typeof a.range||a.range<=0){const e=g.reduce(((e,n)=>Math.min(e,n)),g[0]),n=g.reduce(((e,n)=>Math.max(e,n)),g[0]);a.range=n-e}a.range<=0&&(a.range=1),w=1,A="FLOAT32ARRAY",b=new Float32Array([a.range]),y=`\n range${t}: f32`}const x=g instanceof Int32Array?"i32":g instanceof Float32Array?"f32":"u32",I=`data${t}: array<array<${x}, ${h}>, ${f.length}>`;return{flatSourceArray:g,sourceArraySize:g.length,maxEntryLen:h,arraySizes:d,complexity:m,suppInfoBuffer:b,suppInfoSize:w,suppInfoType:A,suppInfoStructWgsl:y,entryType:u,dataTypeWGSL:x,dataStructWgsl:I,EncodedArrayConstructor:p}}(e,a[n],n,u[n])));if(0===p)throw new Error("No columns provided. Please provide at least one column of data.");1===p&&(s=e.MANHATTAN);let w=y.map((e=>e.suppInfoStructWgsl)).filter((e=>!!e&&""!=e)).join(",\n"),A=!1;w&&""!=w.trim()||(A=!0,w="\ndummy: f32\n");const b=y.map((e=>e.dataStructWgsl)).filter((e=>!!e&&""!=e)).join(",\n"),x=new Uint32Array(p*g);y.forEach(((e,n)=>{x.set(e.arraySizes,n*g)}));const I=1e4,E=100,M=y.reduce(((e,n)=>e+n.complexity),0),N=Math.ceil(1e4/M),v=Math.ceil(Math.sqrt(Math.ceil(100))),C=10*v,B=g*(g-1)/2,S=Math.ceil(B/I),U=c.createShaderModule({label:"Sparse matrix compute shader",code:`\n // each thread will perform 100 iterations at one time, comparing 100 pairs of entries.\n // in total, each thread will perform at most ${S} comparisons.\n // first is the result struct, containing is, js, and distances. each array with length of 100,\n // and also integer for how many pairs were found to be below threshold.\n struct SparseResult {\n i: array<array<u32, 100>, 10000>,\n j: array<array<u32, 100>, 10000>,\n distances: array<array<f32, 100>, 10000>,\n found: array<u32, 10000>,\n done: array<u32, 10000>\n }\n // struct for the data\n struct ComputeInfo {\n // start at cols and rows, and end at cols and rows for each thread, these will be calculated on cpu and passed to gpu.\n startAtCols: array<u32, 10000>,\n startAtRows: array<u32, 10000>,\n endAtCols: array<u32, 10000>,\n endAtRows: array<u32, 10000>,\n\n // the ACTUALLY sizes of each entry\n entrySizes: array<array<u32, ${g}>, ${p}>,\n // the weights for each entry\n weights: array<f32, ${p}>,\n // the data for each entry\n ${b} // an example of the dataWgsl would be:\n //data0: array<array<u32,20>,100>,\n //data1: array<array<u32,20>,100>\n }\n\n // struct for the supplementary information\n struct SuppInfo {\n // struct containing all the supplementary info, like scoring matrix, alphabet indexes, range, etc.\n ${w}\n };\n\n @group(0) @binding(0) var<storage, read_write> computeInfo: ComputeInfo;\n @group(0) @binding(1) var<storage, read_write> suppInfo: SuppInfo;\n @group(0) @binding(2) var<storage, read_write> results: SparseResult;\n @compute @workgroup_size(10, 10) fn calcSparseMatrix(\n @builtin(global_invocation_id) id: vec3<u32>\n ) {\n ${A?"let otherDummy = suppInfo.dummy * 2;":""} // just to make sure that the suppInfo is not optimized out\n let threadCol = id.x;\n let threadRow = id.y;\n let linearIndex = threadRow * ${C} + threadCol;\n if (linearIndex >= 10000) {\n return; // if we are out of bounds, return\n } \n var startAtCol: u32 = computeInfo.startAtCols[linearIndex];\n var startAtRow: u32 = computeInfo.startAtRows[linearIndex];\n let endAtCol: u32 = min(computeInfo.endAtCols[linearIndex], ${g}u);\n let endAtRow: u32 = min(computeInfo.endAtRows[linearIndex], ${g}u);\n let is = &results.i[linearIndex];\n let js = &results.j[linearIndex];\n let distances = &results.distances[linearIndex];\n results.found[linearIndex] = 0; // initialize the found counter\n var found: u32 = 0;\n if (results.done[linearIndex] > 0) {\n return; // if we are done, return\n }\n for (var i = 0; i < ${N}; i++) {\n if (startAtCol >= endAtCol && startAtRow >= endAtRow) {\n results.done[linearIndex] = 1;\n break;\n }\n if (found >= 100) {\n break;\n }\n let dist = combinedDistance(startAtCol, startAtRow);\n if (dist <= ${h}) {\n (*is)[found] = startAtCol;\n (*js)[found] = startAtRow;\n (*distances)[found] = dist;\n found = found + 1;\n }\n startAtCol = startAtCol + 1;\n if (startAtCol >= ${g}u) {\n startAtRow += 1;\n startAtCol = startAtRow + 1;\n }\n }\n results.found[linearIndex] = found;\n // update the startAtCols and startAtRows\n computeInfo.startAtCols[linearIndex] = startAtCol;\n computeInfo.startAtRows[linearIndex] = startAtRow;\n\n }\n\n // this will generate the distance script for each distance metric and then combine them into one\n ${m(a,y.map((e=>e.maxEntryLen)),h,s)}\n\n\n `}),L=c.createComputePipeline({label:"sparse matrix compute pipeline",layout:"auto",compute:{module:U,entryPoint:"calcSparseMatrix"}}),P=new Uint32Array(I),T=new Uint32Array(I),R=new Uint32Array(I),O=new Uint32Array(I),_=Math.floor(B/I);let k=0,D=1;console.time("GPUthreadStarts");for(let e=0;e<I;e++){const n=9999===e?B-1:(e+1)*_,t=g-2-Math.floor(Math.sqrt(-8*n+4*g*(g-1)-7)/2-.5),r=n-g*t+Math.floor((t+1)*(t+2)/2);P[e]=D,T[e]=k,R[e]=r,O[e]=t,k=t,D=r}console.timeEnd("GPUthreadStarts");const j=4e4+g*p+p+y.reduce(((e,n)=>e+n.sourceArraySize),0),$=y.reduce(((e,n)=>e+n.suppInfoSize),0),G=1e6,z=j*Uint32Array.BYTES_PER_ELEMENT;let Y=z;const F=15&z;0!==F&&(Y+=16-F);const H=c.createBuffer({label:"compute info buffer",size:Y,usage:GPUBufferUsage.STORAGE|GPUBufferUsage.COPY_SRC|GPUBufferUsage.COPY_DST,mappedAtCreation:!0}),K=H.getMappedRange();let W=0;new Uint32Array(K,W,I).set(P),W+=I*Uint32Array.BYTES_PER_ELEMENT,new Uint32Array(K,W,I).set(T),W+=I*Uint32Array.BYTES_PER_ELEMENT,new Uint32Array(K,W,I).set(R),W+=I*Uint32Array.BYTES_PER_ELEMENT,new Uint32Array(K,W,I).set(O),W+=I*Uint32Array.BYTES_PER_ELEMENT,new Uint32Array(K,W,x.length).set(x),W+=x.length*Uint32Array.BYTES_PER_ELEMENT,new Float32Array(K,W,p).set(l),W+=p*Float32Array.BYTES_PER_ELEMENT;for(const e of y){const n=e.EncodedArrayConstructor,t=e.sourceArraySize;new n(K,W,t).set(e.flatSourceArray),W+=t*n.BYTES_PER_ELEMENT}H.unmap();const q=$*Uint32Array.BYTES_PER_ELEMENT;let X=q;const V=15&q;0!==V&&(X+=16-V),X=Math.max(X,16);const J=c.createBuffer({label:"supp info buffer",size:X,usage:GPUBufferUsage.STORAGE|GPUBufferUsage.COPY_SRC|GPUBufferUsage.COPY_DST,mappedAtCreation:!0}),Q=J.getMappedRange();let Z=0;for(const e of y)e.suppInfoBuffer&&e.suppInfoBuffer.byteLength>0&&e.suppInfoSize>0&&(new("UINT32ARRAY"===e.suppInfoType?Uint32Array:Float32Array)(Q,Z,e.suppInfoBuffer.length).set(e.suppInfoBuffer),Z+=e.suppInfoBuffer.byteLength);0===Z&&new Uint32Array(Q,0,4).set([1,1,1,1]),J.unmap();const ee=302e4*Uint32Array.BYTES_PER_ELEMENT;let ne=ee;const te=15ⅇ0!==te&&(ne+=16-te);const re=c.createBuffer({label:"results buffer",size:ne,usage:GPUBufferUsage.STORAGE|GPUBufferUsage.COPY_SRC}),ae=c.createBindGroup({label:"bindGroup for sparse matrix buffer",layout:L.getBindGroupLayout(0),entries:[{binding:0,resource:{buffer:H}},{binding:1,resource:{buffer:J}},{binding:2,resource:{buffer:re}}]}),ie=c.createBuffer({label:"results out buffer",size:re.size,usage:GPUBufferUsage.MAP_READ|GPUBufferUsage.COPY_DST}),oe=[],se=[],le=[];let ue=!1;for(;!ue;){const e=c.createCommandEncoder({label:"distance encoder"}),n=e.beginComputePass({label:"distance compute pass"});n.setPipeline(L),n.setBindGroup(0,ae),n.dispatchWorkgroups(v,v),n.end(),e.copyBufferToBuffer(re,0,ie,0,ie.size);const t=e.finish();c.queue.submit([t]),yield c.queue.onSubmittedWorkDone(),yield ie.mapAsync(GPUMapMode.READ);const r=ie.getMappedRange();let a=0;const i=new Uint32Array(r,a,G);a+=G*Uint32Array.BYTES_PER_ELEMENT;const o=new Uint32Array(r,a,G);a+=G*Uint32Array.BYTES_PER_ELEMENT;const s=new Float32Array(r,a,G);a+=G*Float32Array.BYTES_PER_ELEMENT;const l=new Uint32Array(r,a,I);a+=I*Uint32Array.BYTES_PER_ELEMENT,ue=new Uint32Array(r,a,I).every((e=>1===e));const u=l.reduce(((e,n)=>e+n),0),f=new Uint32Array(u),d=new Uint32Array(u),h=new Float32Array(u);let m=0;for(let e=0;e<l.length;e++){const n=l[e];0!==n&&(f.set(i.subarray(e*E,e*E+n),m),d.set(o.subarray(e*E,e*E+n),m),h.set(s.subarray(e*E,e*E+n),m),m+=n)}oe.push(f),se.push(d),le.push(h),ie.unmap()}const fe=oe.reduce(((e,n)=>e+n.length),0),ce=new Uint32Array(fe),de=new Uint32Array(fe),he=new Float32Array(fe);let me=0;for(let e=0;e<oe.length;e++)ce.set(oe[e],me),de.set(se[e],me),he.set(le[e],me),me+=oe[e].length;return H.destroy(),J.destroy(),re.destroy(),ie.destroy(),{i:ce,j:de,distance:he}},new((h=void 0)||(h=Promise))((function(e,n){function t(e){try{a(p.next(e))}catch(e){n(e)}}function r(e){try{a(p.throw(e))}catch(e){n(e)}}function a(n){var a;n.done?e(n.value):(a=n.value,a instanceof h?a:new h((function(e){e(a)}))).then(t,r)}a((p=p.apply(c,d||[])).next())}));var c,d,h,p}function m(e,n,r,i){return e.map(((e,t)=>`\n fn distanceScript${t}(aIndex: u32, bIndex: u32) -> f32 {\n let a = computeInfo.data${t}[aIndex];\n let b = computeInfo.data${t}[bIndex];\n let maxDistance: f32 = ${r};\n ${a[e](n[t],t)}\n }\n `)).join("\n")+"\n"+`\n fn combinedDistance(aIndex: u32, bIndex: u32) -> f32 {\n var distances: array<f32, ${e.length}>;\n ${e.map(((e,n)=>`distances[${n}] = distanceScript${n}(aIndex, bIndex);`)).join("\n")}\n ${t[i](e.length)}\n }\n \n `}(d=c||(c={})).EUCLIDEAN="EUCLIDEAN",d.MANHATTAN="MANHATTAN";const p=e=>null==e;function g(e,n,t,r){if(t>e[e.length-1])return;const a=e.findIndex((e=>t<e));e.pop(),e.splice(a,0,t),n.pop(),n.splice(a,0,r)}class y{constructor(){this._workerCount=Math.max(navigator.hardwareConcurrency-2,1)}async calcMultiColumn(e,t,r,a=[{}],i=[1],o=c.EUCLIDEAN){const s=e[0].length*(e[0].length-1)/2,l=Math.floor(s/this._workerCount),u=e[0].length>2e4?await this.getMinimalThreshold(e,t,a,i,o):0;r<u&&(console.log(`using threshold ${u}`),r=u),a.forEach(((e,n)=>a[n].threshold=r));const f=new Array(this._workerCount),d=new Array(this._workerCount).fill(null).map((()=>new Worker(new URL(n.p+n.u(111),n.b))));for(let n=0;n<this._workerCount;n++)f[n]=new Promise(((u,f)=>{const c=n*l,h=n===this._workerCount-1?s:(n+1)*l;h<=c&&u({i:new Int32Array(0),j:new Int32Array(0),distance:new Float32Array(0),idx:n}),d[n].postMessage({values:e,startIdx:c,endIdx:h,threshold:r,fnNames:t,opts:a,weights:i,aggregationMethod:o}),d[n].onmessage=({data:{error:e,i:t,j:r,distance:a}})=>{e?(d[n].terminate(),f(e)):(d[n].terminate(),u({i:t,j:r,distance:a,idx:n}))}}));const h=await Promise.all(f),m=h.reduce(((e,n)=>e+n.i.length),0),p=new Int32Array(m),g=new Int32Array(m),y=new Float32Array(m);let w=0;for(const e of h)p.set(e.i,w),g.set(e.j,w),y.set(e.distance,w),w+=e.i.length;return{i:p,j:g,distance:y}}async calc(e,n,t,r={}){return await this.calcMultiColumn([e],[n],t,[r],[1])}async getKNN(e,n,t=15,r={}){return await this.multiColumnKNN([e],[n],t,[r],[1])}async getThresholdKNN(e,n,t=.8,r={}){return await this.multiColumnThresholdKnn([e],[n],t,[r],[1])}async multiColumnThresholdKnn(e,t,r=.8,a,i,o=c.EUCLIDEAN){if(e.length!==t.length||e.length!==a.length||e.length!==i.length)throw new Error("values, distance functions, options and weights arrays should have the same length");if(e.some((n=>n.length!==e[0].length)))throw new Error("all values arrays should have the same length");const s=e[0].length*(e[0].length-1)/2,l=Math.floor(s/this._workerCount),u=new Array(this._workerCount),f=new Array(this._workerCount).fill(null).map((()=>new Worker(new URL(n.p+n.u(603),n.b))));for(let n=0;n<this._workerCount;n++)u[n]=new Promise(((u,c)=>{const d=n*l,h=n===this._workerCount-1?s:(n+1)*l;h<=d&&u({knnDistances:new Array(0),knnIndexes:new Array(0)}),f[n].postMessage({values:e,startIdx:d,endIdx:h,fnNames:t,opts:a,threshold:r,weights:i,aggregationMethod:o}),f[n].onmessage=({data:{error:e,knnDistances:t,knnIndexes:r}})=>{e?(f[n].terminate(),c(e)):(f[n].terminate(),u({knnDistances:t,knnIndexes:r}))}}));const d=await Promise.all(u),h=new Int32Array(e[0].length);for(const n of d)for(let t=0;t<e[0].length;++t)h[t]+=n.knnIndexes[t]?.length??0;const m={knnDistances:new Array(e[0].length).fill(null).map(((e,n)=>new Array(h[n]))),knnIndexes:new Array(e[0].length).fill(null).map(((e,n)=>new Array(h[n])))};for(const n of d)for(let t=0;t<e[0].length;++t)for(let e=0;e<n.knnDistances[t]?.length;++e)m.knnDistances[t][h[t]-1]=n.knnDistances[t][e],m.knnIndexes[t][h[t]-1]=n.knnIndexes[t][e],h[t]-=1;return m}async multiColumnKNN(e,t,r=15,a,i,o=c.EUCLIDEAN){if(e.length!==t.length||e.length!==a.length||e.length!==i.length)throw new Error("values, distance functions, options and weights arrays should have the same length");if(e.some((n=>n.length!==e[0].length)))throw new Error("all values arrays should have the same length");const s=e[0].length*(e[0].length-1)/2,l=Math.floor(s/this._workerCount),u=new Array(this._workerCount),f=new Array(this._workerCount).fill(null).map((()=>new Worker(new URL(n.p+n.u(950),n.b))));for(let n=0;n<this._workerCount;n++)u[n]=new Promise(((u,c)=>{const d=n*l,h=n===this._workerCount-1?s:(n+1)*l;h<=d&&u({knnDistances:new Array(0),knnIndexes:new Array(0)}),f[n].postMessage({values:e,startIdx:d,endIdx:h,fnNames:t,opts:a,nNeighbours:r,weights:i,aggregationMethod:o}),f[n].onmessage=({data:{error:e,knnDistances:t,knnIndexes:r}})=>{e?(f[n].terminate(),c(e)):(f[n].terminate(),u({knnDistances:t,knnIndexes:r}))}}));const d=await Promise.all(u),h={knnDistances:new Array(e[0].length).fill(null).map((()=>new Array(r).fill(99999))),knnIndexes:new Array(e[0].length).fill(null).map((()=>new Array(r).fill(-1)))};for(const n of d)for(let t=0;t<e[0].length;++t)for(let e=0;e<n.knnDistances[t]?.length;++e)g(h.knnDistances[t],h.knnIndexes[t],n.knnDistances[t][e],n.knnIndexes[t][e]);return h}async getSampleDistances(e,t,r=[],a,i=c.EUCLIDEAN){const o=new Array(this._workerCount).fill(null).map((()=>new Worker(new URL(n.p+n.u(778),n.b))));try{const n=e[0].length*(e[0].length-1)/2,s=Math.floor(n/this._workerCount),l=1e6,u=Math.max(Math.min(n/1e3,l),Math.min(n,l)),f=Math.floor(u/this._workerCount),c=new Array(this._workerCount);for(let l=0;l<this._workerCount;l++)c[l]=new Promise(((u,c)=>{const d=l*s,h=l===this._workerCount-1?n:(l+1)*s;o[l].postMessage({values:e,startIdx:d,endIdx:h,sampleLength:f,fnNames:t,opts:r,weights:a,aggregationMethod:i}),o[l].onmessage=({data:{error:e,distance:n}})=>{o[l].terminate(),e?c(e):u({distance:n})}}));const d=await Promise.all(c),h=d.reduce(((e,n)=>e+n.distance.length),0),m=new Float32Array(h);let p=0;for(const e of d)m.set(e.distance,p),p+=e.distance.length;return m.sort(),m}catch(e){return o?.forEach((e=>e?.terminate())),console.error(e),new Float32Array(1).fill(.5)}}async getMinimalThreshold(e,n,t=[],r,a=c.EUCLIDEAN){try{const i=e.length*(e.length-1)/2,o=await this.getSampleDistances(e,n,t,r,a);return 1-o[Math.floor(7e7/i*o.length)]}catch(e){return console.error(e),.5}}static calcSync(e,n,t,r){const a=[],i=[],o=[];let s=0,l=0,u=0;const f=e.length*(e.length-1)/2;for(;s<f;){const n=p(e[l])||p(e[u])?1:t(e[l],e[u]);1-n>=r&&(a.push(l),i.push(u),o.push(n)),s++,u++,u===e.length&&(l++,u=l+1)}return{i:new Int32Array(a),j:new Int32Array(i),distance:new Float32Array(o)}}}async function w(e,n,t){const r=e.length,a=[],i=[];for(let o=0;o<r;o++)for(let s=o+1;s<r;s++){const r=e[o],l=e[s];n[r]?.[l]>=t&&(a.push(o),i.push(s))}return function(e,n,t){const r=new Float32Array(t.length).fill(0).map((()=>10*Math.random())),a=new Float32Array(t.length).fill(0).map((()=>10*Math.random())),i=new Float32Array(t.length).fill(0),o=new Float32Array(t.length).fill(0);for(let s=0;s<100;s++){const l=1-s/100;i.fill(0),o.fill(0);for(let t=0;t<e.length;t++){const s=e[t],u=n[t],f=r[s]-r[u],c=a[s]-a[u];Math.abs(f)>=1&&(i[s]-=l*f,i[u]+=l*f),Math.abs(c)>=1&&(o[s]-=l*c,o[u]+=l*c)}for(let e=0;e<t.length;e++){const n=Math.sqrt(i[e]*i[e]+o[e]*o[e]);n>0&&(r[e]+=i[e]/n*l,a[e]+=o[e]/n*l)}}let s=r[0],l=a[0],u=r[0],f=a[0];for(let e=1;e<t.length;e++)s=Math.min(s,r[e]),l=Math.min(l,a[e]),u=Math.max(u,r[e]),f=Math.max(f,a[e]);let c=u-s,d=f-l;0===c&&(c=u);for(let e=0;e<t.length;e++)r[e]=(r[e]-s)/c/2+.5;0===d&&(d=f);for(let e=0;e<t.length;e++)a[e]=(a[e]-l)/d/2+.5;return{embedX:r,embedY:a}}(a,i,e)}function A(e,n,t,r){return a=this,i=void 0,s=function*(){const a=r,i=Math.ceil(a/100),o=Math.ceil(Math.sqrt(i)),s=10*o,l=e.createShaderModule({label:"colwise-normalize",code:`\n @group(0) @binding(0) var<storage, read_write> knnSimilarities : array<f32>;\n @group(0) @binding(1) var<storage, read> offsets : array<u32>;\n @compute @workgroup_size(10, 10) fn normalize(\n @builtin(global_invocation_id) id: vec3<u32>,\n ) {\n let row = id.x;\n let col = id.y;\n let index = row * ${s} + col;\n if (index >= ${r}) {\n return;\n }\n let offsetBegin = offsets[index];\n let offsetEnd = offsets[index + 1];\n var sum = 0.0;\n if (offsetEnd - offsetBegin == 0) {\n return;\n }\n for (var i = offsetBegin; i < offsetEnd; i = i + 1) {\n sum = sum + knnSimilarities[i];\n }\n if (sum > 0.0) {\n for (var i = offsetBegin; i < offsetEnd; i = i + 1) {\n knnSimilarities[i] = knnSimilarities[i] / sum;\n }\n }\n }\n \n `}),u=e.createComputePipeline({label:"hamming compute pipeline",layout:"auto",compute:{module:l,entryPoint:"normalize"}}),f=e.createBuffer({label:"simmilarities info buffer",size:n.byteLength,usage:GPUBufferUsage.STORAGE|GPUBufferUsage.COPY_SRC|GPUBufferUsage.COPY_DST,mappedAtCreation:!0});new Float32Array(f.getMappedRange()).set(n),f.unmap();const c=e.createBuffer({label:"offsets info buffer",size:t.byteLength,usage:GPUBufferUsage.STORAGE|GPUBufferUsage.COPY_SRC|GPUBufferUsage.COPY_DST,mappedAtCreation:!0});new Uint32Array(c.getMappedRange()).set(t),c.unmap();const d=e.createBindGroup({layout:u.getBindGroupLayout(0),entries:[{binding:0,resource:{buffer:f}},{binding:1,resource:{buffer:c}}]}),h=e.createCommandEncoder(),m=h.beginComputePass();m.setPipeline(u),m.setBindGroup(0,d),m.dispatchWorkgroups(o,o),m.end();const p=e.createBuffer({label:"out similarity buffer",size:n.byteLength,usage:GPUBufferUsage.MAP_READ|GPUBufferUsage.COPY_DST});h.copyBufferToBuffer(f,0,p,0,n.byteLength),e.queue.submit([h.finish()]),yield e.queue.onSubmittedWorkDone(),yield p.mapAsync(GPUMapMode.READ);const g=new Float32Array(p.getMappedRange());n.set(g),p.unmap(),f.destroy(),c.destroy(),p.destroy()},new((o=void 0)||(o=Promise))((function(e,n){function t(e){try{l(s.next(e))}catch(e){n(e)}}function r(e){try{l(s.throw(e))}catch(e){n(e)}}function l(n){var a;n.done?e(n.value):(a=n.value,a instanceof o?a:new o((function(e){e(a)}))).then(t,r)}l((s=s.apply(a,i||[])).next())}));var a,i,o,s}function b(e,n=2){for(let t=0;t<e.length;t++)e[t]=Math.pow(e[t],n)}function x(e,n,t,r,a){return i=this,o=void 0,l=function*(){const i=9e4,o=Math.ceil(900),s=Math.ceil(Math.sqrt(o)),l=10*s,u=Math.floor(Math.max(Math.log10(a),2))+1,f=Math.pow(10,-u),c=new Float32Array(n.length),d=e.createShaderModule({label:"expand",code:`\n struct SparseKNN {\n knnSimilarities: array<f32, ${n.length}>,\n knnIndexes: array<u32, ${t.length}>,\n offsets: array<u32, ${r.length}>,\n rowIndexes: array<u32, ${t.length}>,\n }\n\n @group(0) @binding(0) var<storage, read_write> sparseKNN: SparseKNN;\n @group(0) @binding(1) var<storage, read_write> resultSimBlock: array<f32, 90000>;\n @group(0) @binding(2) var<storage, read_write> startAt: u32;\n @compute @workgroup_size(10, 10) fn expand(\n @builtin(global_invocation_id) id: vec3<u32>,\n ) {\n let col = id.x;\n let row = id.y;\n let index = row * ${l} + col;\n if (index >= 90000) {\n return;\n }\n let workingIndex = index + startAt;\n if (workingIndex >= ${t.length}) {\n return;\n }\n \n let rowIdx = sparseKNN.rowIndexes[workingIndex];\n let colIdx = sparseKNN.knnIndexes[workingIndex];\n let offsetBeginRow = sparseKNN.offsets[rowIdx];\n let offsetEndRow = sparseKNN.offsets[rowIdx + 1];\n let offsetBeginCol = sparseKNN.offsets[colIdx];\n let offsetEndCol = sparseKNN.offsets[colIdx + 1];\n var sum = 0.0;\n for (var i = offsetBeginRow; i < offsetEndRow; i = i + 1) {\n for(var j = offsetBeginCol; j < offsetEndCol; j = j + 1) {\n if (sparseKNN.knnIndexes[i] == sparseKNN.knnIndexes[j]) {\n sum = sum + sparseKNN.knnSimilarities[i] * sparseKNN.knnSimilarities[j];\n break;\n }\n }\n }\n if (sum > ${f}) {\n resultSimBlock[index] = sum;\n } else {\n resultSimBlock[index] = 0.0;\n }\n }\n `}),h=e.createComputePipeline({label:"expand compute pipeline",layout:"auto",compute:{module:d,entryPoint:"expand"}});let m=4*(n.length+t.length+r.length+t.length);const p=15&m;0!==p&&(m+=16-p);const g=e.createBuffer({label:"sparse knn buffer",size:m,usage:GPUBufferUsage.STORAGE|GPUBufferUsage.COPY_SRC|GPUBufferUsage.COPY_DST,mappedAtCreation:!0}),y=g.getMappedRange();new Float32Array(y,0,n.length).set(n),new Uint32Array(y,4*n.length,t.length).set(t),new Uint32Array(y,4*(n.length+t.length),r.length).set(r);const w=function(e){const n=new Uint32Array(e[e.length-1]);for(let t=0;t<e.length-1;t++)for(let r=e[t];r<e[t+1];r++)n[r]=t;return n}(r);new Uint32Array(y,4*(n.length+t.length+r.length),w.length).set(w),g.unmap();const A=e.createBuffer({label:"start end buffer",size:4,usage:GPUBufferUsage.STORAGE|GPUBufferUsage.COPY_SRC|GPUBufferUsage.COPY_DST,mappedAtCreation:!0});new Uint32Array(A.getMappedRange()).set([0]),A.unmap();const b=e.createBuffer({label:"result block buffer",size:36e4,usage:GPUBufferUsage.STORAGE|GPUBufferUsage.COPY_SRC|GPUBufferUsage.COPY_DST}),x=e.createBindGroup({layout:h.getBindGroupLayout(0),entries:[{binding:0,resource:{buffer:g}},{binding:1,resource:{buffer:b}},{binding:2,resource:{buffer:A}}]}),I=e.createBuffer({label:"out block buffer",size:b.size,usage:GPUBufferUsage.MAP_READ|GPUBufferUsage.COPY_DST});for(let n=0;n<Math.ceil(t.length/i);n++){const r=n*i,a=Math.min(i,t.length-r);e.queue.writeBuffer(A,0,new Uint32Array([r]));const o=e.createCommandEncoder({label:"expand encoder"}),l=o.beginComputePass({label:"expand compute pass"});l.setPipeline(h),l.setBindGroup(0,x),l.dispatchWorkgroups(s,s),l.end(),o.copyBufferToBuffer(b,0,I,0,I.size),e.queue.submit([o.finish()]),yield e.queue.onSubmittedWorkDone(),yield I.mapAsync(GPUMapMode.READ);const u=new Float32Array(I.getMappedRange(),0,a);c.set(u,r),I.unmap()}return g.destroy(),A.destroy(),b.destroy(),I.destroy(),{KNNIndexes:t,KNNSimilarities:c,indexOffsets:r}},new((s=void 0)||(s=Promise))((function(e,n){function t(e){try{a(l.next(e))}catch(e){n(e)}}function r(e){try{a(l.throw(e))}catch(e){n(e)}}function a(n){var a;n.done?e(n.value):(a=n.value,a instanceof s?a:new s((function(e){e(a)}))).then(t,r)}a((l=l.apply(i,o||[])).next())}));var i,o,s,l}function I(e,n){const t=function(e,n){const t=new Uint32Array(n);for(let n=0;n<e.i.length;n++)t[e.i[n]]++,t[e.j[n]]++;for(let e=0;e<n;e++)t[e]++;return t}(e,n),r=function(e){const n=new Uint32Array(e.length+1);let t=0;for(let r=0;r<e.length;r++)n[r]=t,t+=e[r];return n[e.length]=t,n}(t),a=new Uint32Array(n).fill(1),i=new Uint32Array(2*e.i.length+n),o=new Float32Array(2*e.i.length+n);for(let e=0;e<n;e++)i[r[e]]=e,o[r[e]]=1;for(let n=0;n<e.i.length;n++){const t=e.i[n],s=e.j[n],l=1-e.distance[n];i[r[t]+a[t]]=s,o[r[t]+a[t]]=l,a[t]++,i[r[s]+a[s]]=t,o[r[s]+a[s]]=l,a[s]++}return{KNNIndexes:i,KNNSimilarities:o,indexOffsets:r}}Object.prototype.toString;const E={expandFactor:2,maxIterations:5,inflateFactor:2,multFactor:1};class M{constructor(e={}){this._options={...E,...e}}async transform(e,n){let t=this.toObjectForm(e);if(this._options.maxIterations>0){this.addLoops(t,n),this.normalize(t);for(let e=0;e<this._options.maxIterations;e++)t=this.expand(t,n),this.inflate(t),this.normalize(t)}const{clusters:r,is:a,js:i}=this.assignClusters(t,n);this.correctClusters(r);const o=await this.layout(r,t,n);return{clusters:r,embedX:o.embedX,embedY:o.embedY,is:a,js:i}}async transformWebGPU(e,n){if(0===this._options.maxIterations)return this.transform(e,n);const t=await function(e,n,t=5,r=2){return a=this,i=void 0,s=function*(){const a=yield f();if(!a)throw new Error("no gpu device found");const i=I(e,n);yield A(a,i.KNNSimilarities,i.indexOffsets,n);let o=i;for(let e=0;e<t;e++){const e=yield x(a,o.KNNSimilarities,o.KNNIndexes,o.indexOffsets,n);b(e.KNNSimilarities,r),yield A(a,e.KNNSimilarities,e.indexOffsets,n),o=e}return o},new((o=void 0)||(o=Promise))((function(e,n){function t(e){try{l(s.next(e))}catch(e){n(e)}}function r(e){try{l(s.throw(e))}catch(e){n(e)}}function l(n){var a;n.done?e(n.value):(a=n.value,a instanceof o?a:new o((function(e){e(a)}))).then(t,r)}l((s=s.apply(a,i||[])).next())}));var a,i,o,s}(e,n,this._options.maxIterations,this._options.inflateFactor),r=this.csrToSparseObject(t,n),{clusters:a,is:i,js:o}=this.assignClusters(r,n);this.correctClusters(a);const s=await this.layout(a,r,n);return{clusters:a,embedX:s.embedX,embedY:s.embedY,is:i,js:o}}correctClusters(e){const n={};for(const t of e)n[t]||(n[t]=0),n[t]++;const t=Object.keys(n).map(Number).sort(((e,t)=>n[t]-n[e])),r={};t.forEach(((e,n)=>r[e]=n+1));for(let n=0;n<e.length;n++)e[n]=r[e[n]]}csrToSparseObject(e,n){const t=Math.floor(Math.max(Math.log10(n),2))+1,r=1/Math.pow(10,t),a={};for(let t=0;t<n;t++){a[t]={};for(let n=e.indexOffsets[t];n<e.indexOffsets[t+1];n++){const i=e.KNNIndexes[n];i<=t||e.KNNSimilarities[n]<r||(a[t][i]=e.KNNSimilarities[n])}}return a}async layout(e,n,t){const r=new Float32Array(t).fill(0),a=new Float32Array(t).fill(0),i={};e.forEach(((e,n)=>{i[e]||(i[e]=[]),i[e].push(n)}));let o=0;const s=Object.keys(i);s.sort(((e,n)=>i[n].length-i[e].length));let l=6,u=0;for(const e of s){const t=i[e],s=await w(t,n,.001);o===Math.ceil(l/1.5)&&(o=0,u+=5/l,l=Math.ceil(1.5*l));const f=o%l*5/l*1.5;for(let e=0;e<s.embedX.length;e++)r[t[e]]=5*s.embedX[e]/l+f,a[t[e]]=5*s.embedY[e]/l+u;o++}return{embedX:r,embedY:a}}mergeClusters(e,n,t){const r=e[n],a=e[t];for(let n=0;n<e.length;n++)e[n]===a&&(e[n]=r)}assignClusters(e,n){let t=0;const r=[],a=[],i=Math.floor(Math.max(Math.log10(n),2))+1,o=Math.pow(10,i),s=new Array(n).fill(-1);for(const n of Object.keys(e))for(const i of Object.keys(e[n]))Math.round(e[n][i]*o)/o>0&&e[n][i]!==Number(n)&&Number(i)>Number(n)&&(r.push(Number(n)),a.push(Number(i)),-1!==s[Number(n)]&&-1!==s[Number(i)]?s[Number(n)]!==s[Number(i)]&&this.mergeClusters(s,Number(n),Number(i)):-1!==s[Number(n)]?s[Number(i)]=s[Number(n)]:-1!==s[Number(i)]?s[Number(n)]=s[Number(i)]:(t++,s[Number(n)]=t,s[Number(i)]=t));for(let e=0;e<s.length;e++)-1===s[e]&&(t++,s[e]=t);return{clusters:s,is:new Uint32Array(r),js:new Uint32Array(a)}}assignClustersCSR(e,n){let t=0;const r=[],a=[],i=Math.floor(Math.max(Math.log10(n),2))+1,o=1/Math.pow(10,i),s=new Array(n).fill(-1),l=new Uint32Array(n+1);let u=0;l[0]=0;for(let i=0;i<n;i++){for(let n=e.indexOffsets[i];n<e.indexOffsets[i+1];n++){const l=e.KNNIndexes[n];l<=i||e.KNNSimilarities[n]<=o||(r.push(i),a.push(l),u++,-1!==s[i]&&-1!==s[l]?s[i]!==s[l]&&this.mergeClusters(s,i,l):-1!==s[i]?s[l]=s[i]:-1!==s[l]?s[i]=s[l]:(t++,s[i]=t,s[l]=t))}l[i+1]=u}for(let e=0;e<s.length;e++)-1===s[e]&&(t++,s[e]=t);return{clusters:s,is:new Uint32Array(r),js:new Uint32Array(a),correctedOffsets:l}}toObjectForm(e){const n={};for(let t=0;t<e.i.length;t++)n[e.i[t]]||(n[e.i[t]]={}),n[e.i[t]][e.j[t]]=1-e.distance[t],n[e.j[t]]||(n[e.j[t]]={}),n[e.j[t]][e.i[t]]=1-e.distance[t];return n}addLoops(e,n){for(let t=0;t<n;t++)e[t]||(e[t]={}),e[t][t]=this._options.multFactor}normalize(e){for(const n of Object.keys(e)){const t=e[n];let r=0;for(const e of Object.keys(t))r+=t[e];if(0!==r)for(const a of Object.keys(t))e[n][a]/=r}}expand(e,n){const t={},r=Math.floor(Math.max(Math.log10(n),2))+1,a=Math.pow(10,r);for(let r=0;r<n;r++)if(e[r]){t[r]??(t[r]={});for(let i=r;i<n;i++){if(!e[r]?.[i])continue;const n=this.getExpandValue(e,r,i);Math.round(n*a)/a>0&&(t[r][i]=n,t[i]||(t[i]={}),t[i][r]=n)}}return t}inflate(e){for(const n of Object.keys(e)){const t=e[n];for(const r of Object.keys(t))e[n][r]=Math.pow(e[n][r],this._options.inflateFactor)}}getExpandValue(e,n,t){let r=0;const a=Object.keys(e[n]??{}),i=Object.keys(e[t]??{});for(const o of a)i.includes(o)&&(r+=e[n][o]*e[t][o]);return r}}onmessage=async e=>{const{data:n,threshold:t,weights:r,aggregationMethod:a,distanceFnArgs:i,distanceFns:o,maxIterations:s,useWebGPU:l,inflate:u}=e.data;console.time("sparse matrix");let f=null;if(l)try{f=await h(n,t/100,o,a,r,i)}catch(e){console.error(e)}f||(l&&console.error("WEBGPU sparse matrix calculation failed, falling back to CPU implementation"),f=await(new y).calcMultiColumn(n,o,t/100,i,r,a)),console.timeEnd("sparse matrix");const c=new M({maxIterations:s??5,inflateFactor:u??2});console.time("MCL");let d=null;if(l)try{d=await c.transformWebGPU(f,n[0].length)}catch(e){console.error("webGPU MCL failed, falling back to CPU implementation"),console.error(e)}d||(d=await c.transform(f,n[0].length)),console.timeEnd("MCL"),postMessage({res:d})},peptides={}})();
|
|
2
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"795.js","mappings":"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","sources":["webpack://peptides/./node_modules/@datagrok-libraries/math/src/webGPU/multi-col-distances/webGPU-aggregation.js","webpack://peptides/webpack/bootstrap","webpack://peptides/webpack/runtime/get javascript chunk filename","webpack://peptides/webpack/runtime/global","webpack://peptides/webpack/runtime/hasOwnProperty shorthand","webpack://peptides/webpack/runtime/publicPath","webpack://peptides/webpack/runtime/importScripts chunk loading","webpack://peptides/./node_modules/@datagrok-libraries/math/src/webGPU/multi-col-distances/webGPU-multicol-distances.js","webpack://peptides/./node_modules/@datagrok-libraries/math/src/webGPU/getGPUDevice.js","webpack://peptides/./node_modules/@datagrok-libraries/math/src/webGPU/sparse-matrix/webGPU-sparse-matrix.js","webpack://peptides/./node_modules/@datagrok-libraries/ml/src/distance-matrix/types.js","webpack://peptides/./node_modules/@datagrok-libraries/math/src/webGPU/preprocessing/webGPU-process-info.js","webpack://peptides/./node_modules/@datagrok-libraries/ml/src/distance-matrix/utils.js","webpack://peptides/./node_modules/@datagrok-libraries/ml/src/distance-matrix/sparse-matrix-service.js","webpack://peptides/./node_modules/@datagrok-libraries/ml/src/MCL/bio-layout.js","webpack://peptides/./node_modules/@datagrok-libraries/math/src/webGPU/MCL/colwise-normalize.js","webpack://peptides/./node_modules/@datagrok-libraries/math/src/webGPU/MCL/inflate-expand.js","webpack://peptides/./node_modules/@datagrok-libraries/math/src/webGPU/MCL/utils.js","webpack://peptides/./node_modules/@datagrok-libraries/math/src/webGPU/umap/utils.js","webpack://peptides/./node_modules/is-any-array/lib-esm/index.js","webpack://peptides/./node_modules/@datagrok-libraries/ml/src/MCL/marcov-cluster.js","webpack://peptides/./node_modules/@datagrok-libraries/math/src/webGPU/MCL/MCL-webGPU.js","webpack://peptides/./node_modules/@datagrok-libraries/ml/src/MCL/mcl-worker.js"],"sourcesContent":["function euclideanAggregationWgsl(arraySize) {\n    return `\n        var sum = 0.0;\n        for (var i = 0u; i < ${arraySize}; i = i + 1u) {\n            sum = sum + distances[i] * distances[i] * computeInfo.weights[i] * computeInfo.weights[i];\n        }\n        return sqrt(sum);\n    `;\n}\n;\nfunction manhattanAggregationWgsl(arraySize) {\n    return `\n        var sum = 0.0;\n        for (var i = 0u; i < ${arraySize}; i = i + 1u) {\n            sum = sum + abs(distances[i]) * computeInfo.weights[i];\n        }\n        return sum;\n    `;\n}\nexport var WEBGSLAGGREGATION;\n(function (WEBGSLAGGREGATION) {\n    WEBGSLAGGREGATION[\"EUCLIDEAN\"] = \"EUCLIDEAN\";\n    WEBGSLAGGREGATION[\"MANHATTAN\"] = \"MANHATTAN\";\n})(WEBGSLAGGREGATION || (WEBGSLAGGREGATION = {}));\nexport const WEBGSLAGGREGATIONFUNCTIONS = {\n    [WEBGSLAGGREGATION.EUCLIDEAN]: euclideanAggregationWgsl,\n    [WEBGSLAGGREGATION.MANHATTAN]: manhattanAggregationWgsl\n};\n//# sourceMappingURL=webGPU-aggregation.js.map","// The require scope\nvar __webpack_require__ = {};\n\n// expose the modules object (__webpack_modules__)\n__webpack_require__.m = __webpack_modules__;\n\n","// This function allow to reference async chunks\n__webpack_require__.u = (chunkId) => {\n\t// return url for filenames based on template\n\treturn \"\" + chunkId + \".js\";\n};","__webpack_require__.g = (function() {\n\tif (typeof globalThis === 'object') return globalThis;\n\ttry {\n\t\treturn this || new Function('return this')();\n\t} catch (e) {\n\t\tif (typeof window === 'object') return window;\n\t}\n})();","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","var scriptUrl;\nif (__webpack_require__.g.importScripts) scriptUrl = __webpack_require__.g.location + \"\";\nvar document = __webpack_require__.g.document;\nif (!scriptUrl && document) {\n\tif (document.currentScript)\n\t\tscriptUrl = document.currentScript.src;\n\tif (!scriptUrl) {\n\t\tvar scripts = document.getElementsByTagName(\"script\");\n\t\tif(scripts.length) {\n\t\t\tvar i = scripts.length - 1;\n\t\t\twhile (i > -1 && (!scriptUrl || !/^http(s?):/.test(scriptUrl))) scriptUrl = scripts[i--].src;\n\t\t}\n\t}\n}\n// When supporting browsers where an automatic publicPath is not supported you must specify an output.publicPath manually via configuration\n// or pass an empty string (\"\") and set the __webpack_public_path__ variable from your code to use your own logic.\nif (!scriptUrl) throw new Error(\"Automatic publicPath is not supported in this browser\");\nscriptUrl = scriptUrl.replace(/#.*$/, \"\").replace(/\\?.*$/, \"\").replace(/\\/[^\\/]+$/, \"/\");\n__webpack_require__.p = scriptUrl;","__webpack_require__.b = self.location + \"\";\n\n// object to store loaded chunks\n// \"1\" means \"already loaded\"\nvar installedChunks = {\n\t795: 1\n};\n\n// no chunk install function needed\n// no chunk loading\n\n// no HMR\n\n// no HMR manifest","/* eslint-disable max-len */\n// in all the functions below, the variables a and b are assumed to be arrays of uint32/f32\n//values which are infered from the code this chunk is injected into\n// also, we have access to computeInfo struct, which contains the following fields:\n// computeInfo.entrySizes: array of arrays of u32 containing the sizes of the entries\n// other fields are specific to the distance function should be injected from the main script that calls this function,\n// and should be available in the supplementaryInfo struct\n// like the similarity matrix for monomer chemical distance.\n// the getProcessInfo function should return correct buffer allocation mechanism for the supplementaryInfo,\n// for every entry list\n// the maxDistance variable is also assumed to be available in the\n// scope of the function, in case of knn it is the distance in the last postion of knn on this index,\n// in case of sparse matrix, it can be just the threshold for the distance.\n// hamming distance for sequnences of uint32 arrays of max length ${maxArraySize}\nexport function webGPUHamming(_maxArraySize, entryIndex) {\n    return `\n  let aLength: u32 = computeInfo.entrySizes[${entryIndex}][aIndex];\n  let bLength: u32 = computeInfo.entrySizes[${entryIndex}][bIndex];\n  let maxLength: u32 = max(aLength, bLength);\n  let minLength: u32 = min(aLength, bLength);\n  let sizeDiff: u32 = maxLength - minLength;\n  \n  let maxIntDistance = ceil(maxDistance * f32(maxLength)) - f32(sizeDiff);\n\n  var diff: f32 = 0.0;\n  for (var i = 0u; i < ${_maxArraySize}; i = i + 1u) {\n      diff = diff + f32(a[i] != b[i]);\n      if (diff > maxIntDistance) {\n            return 1.0;\n        }\n  }\n  diff += f32(sizeDiff);\n  return diff / ${_maxArraySize};\n  `;\n}\nexport function webGPUMonomerChemicalDistance(_maxArraySize, entryIndex) {\n    // it is assumet that suppInfo struct contains correct matrix called similarityMatrix${entryIndex}, (similarityMatrix0, similarityMatrix1, etc)\n    // this should be guaranteed by the getProcessInfo function.\n    return `\n  let aLength: u32 = computeInfo.entrySizes[${entryIndex}][aIndex];\n  let bLength: u32 = computeInfo.entrySizes[${entryIndex}][bIndex];\n  let maxLength: u32 = max(aLength, bLength);\n  let minLength: u32 = min(aLength, bLength);\n  let sizeDiff: u32 = maxLength - minLength;\n  \n  let maxIntDistance = ceil(maxDistance * f32(maxLength)) - f32(sizeDiff);\n\n  let simMatrix = &(suppInfo.similarityMatrix${entryIndex}); // using pointers make things faster\n  var diff: f32 = 0.0;\n  for (var i = 0u; i < ${_maxArraySize}; i = i + 1u) {\n      diff = diff + 1.0 - (*simMatrix)[u32(a[i])][u32(b[i])];\n      if (diff > maxIntDistance) {\n            return 1.0;\n        }\n  }\n  diff += f32(sizeDiff);\n  return diff / ${_maxArraySize};\n  `;\n}\nexport function webGPULevenstein(maxArraySize, entryIndex) {\n    return `\n  let aLength: u32 = computeInfo.entrySizes[${entryIndex}][aIndex];\n  let bLength: u32 = computeInfo.entrySizes[${entryIndex}][bIndex];\n  let maxLength: u32 = max(aLength, bLength);\n  let minLength: u32 = min(aLength, bLength);\n\n  let maxIntDistance = ceil(maxDistance * f32(maxLength));\n\n  // we will store two arrays as matrix and swap the working indices per pass.\n  // this way we can reduce memory usage per computation to just O(aLength)\n  // the grid will have aLength + 1 columns and bLength + 1 rows\n  // this will be guaranteed by iteration, but the array sizes must be known at compile time, so we will use a fixed size of maxArraySize\n  var dynamicPassMat: array<array<f32, ${maxArraySize + 1}u>, 2>; // initialize to 0\n  \n  var prevIndex: u32 = 0;\n  var curIndex: u32 = 1; // we will swap these indices per pass\n\n  // initialize the first row\n  for (var i = 0u; i <= aLength; i = i + 1u) {\n      dynamicPassMat[prevIndex][i] = f32(i);\n  }\n\n  // iterate over the rows\n  for (var i = 1u; i <= bLength; i = i + 1u) {\n      dynamicPassMat[curIndex][0] = f32(i);\n      var minEntry: f32 = f32(maxLength);\n      let prevRow = &dynamicPassMat[prevIndex];\n      let curRow = &dynamicPassMat[curIndex];\n      let bMon = u32(b[i - 1]);\n      for (var j = 1u; j <= aLength; j = j + 1u) {\n          var cost: f32 = f32(a[j - 1] != bMon);\n          var res: f32 = min(\n              min(\n                (*prevRow)[j] + 1.0, // deletion\n                (*curRow)[j - 1] + 1.0, // insertion\n              ),\n              (*prevRow)[j - 1] + cost // substitution\n              );\n              (*curRow)[j] = res;\n          if (res < minEntry) {\n              minEntry = res;\n          }\n      }\n      // swap the indices\n      let temp: u32 = prevIndex;\n      prevIndex = curIndex;\n      curIndex = temp;\n      if (minEntry > maxIntDistance) {\n          return 1.0;\n      }\n  }\n\n  return dynamicPassMat[prevIndex][aLength] / f32(maxLength);\n  `;\n}\nexport function webGPUNeedlemanWunsch(maxArraySize, entryIndex) {\n    // version of the levenshtain where the cost of substitution is customizable\n    // it is assumet that suppInfo struct contains correct matrix called similarityMatrix${entryIndex}, (similarityMatrix0, similarityMatrix1, etc)\n    // and gapOpenPenalty, gapExtensionPenalty\n    // this should be guaranteed by the getProcessInfo function.\n    return `\n      let aLength: u32 = computeInfo.entrySizes[${entryIndex}][aIndex];\n      let bLength: u32 = computeInfo.entrySizes[${entryIndex}][bIndex];\n      let maxLength: u32 = max(aLength, bLength);\n      let minLength: u32 = min(aLength, bLength);\n  \n      let maxIntDistance = ceil(maxDistance * f32(maxLength));\n      // we will store two arrays as matrix and swap the working indices per pass.\n      // this way we can reduce memory usage per computation to just O(aLength)\n      // the grid will have aLength + 1 columns and bLength + 1 rows\n      // this will be guaranteed by iteration, but the array sizes must be known at compile time, so we will use a fixed size of maxArraySize\n      var dynamicPassMat: array<array<f32, ${maxArraySize + 1}u>, 2>; // initialize to 0\n      \n      // we need to keep track of which operation led to the current cell\n      // i.e. whether we came from the left, top or diagonal to assign gap open/gap extend penalty\n      var verticalGaps: array<u32, ${maxArraySize + 1}u>;\n      var horizontalGaps: array<u32, ${maxArraySize + 1}u>;\n\n      let gapOpenPenalty: f32 = suppInfo.gapOpenPenalty${entryIndex};\n      let gapExtensionPenalty: f32 = suppInfo.gapExtensionPenalty${entryIndex};\n      var prevIndex: u32 = 0;\n      var curIndex: u32 = 1; // we will swap these indices per pass\n      // initialize the first row\n      for (var i = 0u; i <= aLength; i = i + 1u) {\n          dynamicPassMat[prevIndex][i] = gapOpenPenalty + f32(i - 1) * gapExtensionPenalty;\n          dynamicPassMat[curIndex][i] = 0.0;\n      }\n      dynamicPassMat[0][0] = 0.0;\n\n      let simMatrix = &suppInfo.similarityMatrix${entryIndex}; // using pointers make things faster\n      // iterate over the rows\n      for (var i = 1u; i <= bLength; i = i + 1u) {\n          let prevRow = &dynamicPassMat[prevIndex];\n          let curRow = &dynamicPassMat[curIndex];\n          (*curRow)[0] = gapOpenPenalty + f32(i - 1) * gapExtensionPenalty;\n          var minEntry: f32 = f32(maxLength);\n          let monB = u32(b[i - 1]);\n          for (var j = 1u; j <= aLength; j = j + 1u) {\n              let monA = u32(a[j - 1]);\n              \n              let cost: f32 = (*prevRow)[j - 1] + 1f - (*simMatrix)[monA][monB];\n              var top = (*prevRow)[j]; // deletion\n              if (verticalGaps[j] > 0) {\n                  top = top + gapExtensionPenalty;\n              } else {\n                  top = top + gapOpenPenalty;\n              }\n              var left = (*curRow)[j - 1]; // insertion\n              if (horizontalGaps[j - 1] > 0) {\n                  left = left + gapExtensionPenalty;\n              } else {\n                  left = left + gapOpenPenalty;\n              }\n              var res: f32 = min(\n                  min(\n                      top, // deletion\n                      left, // insertion\n                  ),\n                  cost // substitution\n                  );\n                  (*curRow)[j] = res;\n              if (res < minEntry) {\n                  minEntry = res;\n              }\n              // update the horizontal and vertical gaps\n              if (res == cost) {\n                  verticalGaps[j] = 0;\n                  horizontalGaps[j] = 0;\n              } else if (res == left) {\n                  verticalGaps[j] = 0;\n                  horizontalGaps[j] = 1;\n              } else {\n                  verticalGaps[j] = 1;\n                  horizontalGaps[j] = 0;\n              }\n          }\n          // swap the indices\n          let temp: u32 = prevIndex;\n          prevIndex = curIndex;\n          curIndex = temp;\n          if (minEntry > maxIntDistance) {\n              return 1.0;\n          }\n      }\n      return dynamicPassMat[prevIndex][aLength] / f32(maxLength);\n  \n  `;\n}\nexport function webGPUEuclidean(maxArraySize, _entryIndex) {\n    return `\n  var dist: f32 = 0.0;\n  for (var i = 0u; i < ${maxArraySize}; i = i + 1u) {\n    dist = dist + f32(a[i] - b[i]) * f32(a[i] - b[i]);\n  }\n  return sqrt(dist);\n  `;\n}\nexport function webGPUManhattan(maxArraySize, _entryIndex) {\n    return `\n  var dist: f32 = 0.0;\n  for (var i = 0u; i < ${maxArraySize}; i = i + 1u) {\n    dist = dist + abs(f32(a[i] - b[i]));\n  }\n  return dist;\n  `;\n}\nexport function webGPUOneHotDistance(_maxArraySize, entryIndex) {\n    return `\n  let aLength: u32 = computeInfo.entrySizes[${entryIndex}][aIndex];\n  let bLength: u32 = computeInfo.entrySizes[${entryIndex}][bIndex];\n  if (aLength != bLength) {\n    return 1.0;\n  }\n  for (var i = 0u; i < aLength; i = i + 1u) {\n    if(a[i] != b[i]) {\n      return 1.0;\n    }\n  }\n  return 0.0;\n  `;\n}\nexport function webGPUNumericDistance(_maxArraySize, entryIndex) {\n    // we assume that range${entryIndex} is available in the supplementaryInfo struct\n    return `\n    let range = suppInfo.range${entryIndex};\n    return f32(abs(f32(a[0]) - f32(b[0])) / range);\n  `;\n}\n// tanimoto distance for uint32 arrays of length ${maxArraySize}\nexport function webGPUTanimotoBitArray(maxArraySize, _entryIndex) {\n    return `\n  var onBitsa: u32 = 0u;\n  var onBitsb: u32 = 0u;\n  for (var i = 0u; i < ${maxArraySize}u; i = i + 1u) {\n      onBitsa = onBitsa + countOneBits(a[i]);\n      onBitsb = onBitsb + countOneBits(b[i]);\n  }\n\n  if (onBitsa == 0u && onBitsb == 0u) {\n      return 0.0;\n  }\n\n  let totalOnBits = onBitsa + onBitsb;\n  var commonBits: u32 = 0u;\n  for (var i = 0u; i < ${maxArraySize}u; i = i + 1u) {\n      commonBits = commonBits + countOneBits(a[i] & b[i]);\n  }\n\n  return 1.0 - f32(commonBits) / f32(totalOnBits - commonBits);\n  `;\n}\nexport function webGPUAsymmetricBitArray(maxArraySize, _entryIndex) {\n    return `\n  var onBitsa: u32 = 0u;\n  var onBitsb: u32 = 0u;\n  for (var i = 0u; i < ${maxArraySize}u; i = i + 1u) {\n      onBitsa = onBitsa + countOneBits(a[i]);\n      onBitsb = onBitsb + countOneBits(b[i]);\n  }\n  let min = min(onBitsa, onBitsb);\n  if (min == 0u) {\n    return 1.0;\n  }\n  var commonBits: u32 = 0u;\n  for (var i = 0u; i < ${maxArraySize}u; i = i + 1u) {\n      commonBits = commonBits + countOneBits(a[i] & b[i]);\n  }\n  return 1.0 - f32(commonBits) / f32(min);\n  `;\n}\nexport function webGPUCosineBitArray(maxArraySize, _entryIndex) {\n    return `\n  var onBitsa: u32 = 0u;\n  var onBitsb: u32 = 0u;\n  for (var i = 0u; i < ${maxArraySize}u; i = i + 1u) {\n      onBitsa = onBitsa + countOneBits(a[i]);\n      onBitsb = onBitsb + countOneBits(b[i]);\n  }\n  let total = onBitsa * onBitsb; // p.s. here total is taken by multiplying\n  if (total == 0u) {\n    return 1.0;\n  }\n  var commonBits: u32 = 0u;\n  for (var i = 0u; i < ${maxArraySize}u; i = i + 1u) {\n      commonBits = commonBits + countOneBits(a[i] & b[i]);\n  }\n  return 1.0 - f32(commonBits) / sqrt(f32(total));\n  `;\n}\nexport function webGPUSokalBitArray(maxArraySize, _entryIndex) {\n    return `\n  var onBitsa: u32 = 0u;\n  var onBitsb: u32 = 0u;\n  for (var i = 0u; i < ${maxArraySize}u; i = i + 1u) {\n      onBitsa = onBitsa + countOneBits(a[i]);\n      onBitsb = onBitsb + countOneBits(b[i]);\n  }\n  let total = onBitsa + onBitsb;\n  if (total == 0u) {\n    return 1.0;\n  }\n  var commonBits: u32 = 0u;\n  for (var i = 0u; i < ${maxArraySize}u; i = i + 1u) {\n      commonBits = commonBits + countOneBits(a[i] & b[i]);\n  }\n  return 1.0 - f32(commonBits) / f32(total * 2 - commonBits * 3);\n  `;\n}\nexport var WEBGPUDISTANCE;\n(function (WEBGPUDISTANCE) {\n    WEBGPUDISTANCE[\"HAMMING\"] = \"Hamming\";\n    WEBGPUDISTANCE[\"EUCLIDEAN\"] = \"Euclidean\";\n    WEBGPUDISTANCE[\"MANHATTAN\"] = \"Manhattan\";\n    WEBGPUDISTANCE[\"TANIMOTO\"] = \"Tanimoto\";\n    WEBGPUDISTANCE[\"LEVENSTEIN\"] = \"Levenshtein\";\n    WEBGPUDISTANCE[\"NEEDLEMAN_WUNSCH\"] = \"Needlemann-Wunsch\";\n    WEBGPUDISTANCE[\"MONOMER_CHEMICAL_DISTANCE\"] = \"Monomer chemical distance\";\n    WEBGPUDISTANCE[\"SOKAL\"] = \"Sokal\";\n    WEBGPUDISTANCE[\"COSINE\"] = \"Cosine\";\n    WEBGPUDISTANCE[\"ASYMMETRIC\"] = \"Asymmetric\";\n    WEBGPUDISTANCE[\"Difference\"] = \"Difference\";\n    WEBGPUDISTANCE[\"OneHot\"] = \"One-Hot\";\n})(WEBGPUDISTANCE || (WEBGPUDISTANCE = {}));\nexport const webGPUFunctions = {\n    [WEBGPUDISTANCE.HAMMING]: webGPUHamming,\n    [WEBGPUDISTANCE.EUCLIDEAN]: webGPUEuclidean,\n    [WEBGPUDISTANCE.MANHATTAN]: webGPUManhattan,\n    [WEBGPUDISTANCE.TANIMOTO]: webGPUTanimotoBitArray,\n    [WEBGPUDISTANCE.LEVENSTEIN]: webGPULevenstein,\n    [WEBGPUDISTANCE.NEEDLEMAN_WUNSCH]: webGPUNeedlemanWunsch,\n    [WEBGPUDISTANCE.MONOMER_CHEMICAL_DISTANCE]: webGPUMonomerChemicalDistance,\n    [WEBGPUDISTANCE.SOKAL]: webGPUSokalBitArray,\n    [WEBGPUDISTANCE.COSINE]: webGPUCosineBitArray,\n    [WEBGPUDISTANCE.ASYMMETRIC]: webGPUAsymmetricBitArray,\n    [WEBGPUDISTANCE.Difference]: webGPUNumericDistance,\n    [WEBGPUDISTANCE.OneHot]: webGPUOneHotDistance\n};\nexport const distanceFunctionComplexity = {\n    [WEBGPUDISTANCE.HAMMING]: (maxEntrySize) => Math.ceil(maxEntrySize / 30),\n    [WEBGPUDISTANCE.EUCLIDEAN]: (maxEntrySize) => Math.ceil(maxEntrySize / 30),\n    [WEBGPUDISTANCE.MANHATTAN]: (maxEntrySize) => Math.ceil(maxEntrySize / 30),\n    [WEBGPUDISTANCE.TANIMOTO]: (maxEntrySize) => Math.ceil(maxEntrySize / 60),\n    [WEBGPUDISTANCE.SOKAL]: (maxEntrySize) => Math.ceil(maxEntrySize / 60),\n    [WEBGPUDISTANCE.COSINE]: (maxEntrySize) => Math.ceil(maxEntrySize / 60),\n    [WEBGPUDISTANCE.ASYMMETRIC]: (maxEntrySize) => Math.ceil(maxEntrySize / 60),\n    [WEBGPUDISTANCE.LEVENSTEIN]: (maxEntrySize) => Math.ceil(maxEntrySize * maxEntrySize / 60),\n    [WEBGPUDISTANCE.NEEDLEMAN_WUNSCH]: (maxEntrySize) => Math.ceil(maxEntrySize * maxEntrySize / 60),\n    [WEBGPUDISTANCE.MONOMER_CHEMICAL_DISTANCE]: (maxEntrySize) => Math.ceil(maxEntrySize / 25),\n    [WEBGPUDISTANCE.Difference]: (_maxEntrySize) => 1,\n    [WEBGPUDISTANCE.OneHot]: (_maxEntrySize) => Math.ceil(_maxEntrySize / 40),\n};\nexport const TypeSupportedDistances = {\n    [\"STRING\" /* WGPUENTRYTYPE.STRING */]: new Set([WEBGPUDISTANCE.HAMMING, WEBGPUDISTANCE.LEVENSTEIN, WEBGPUDISTANCE.NEEDLEMAN_WUNSCH, WEBGPUDISTANCE.MONOMER_CHEMICAL_DISTANCE, WEBGPUDISTANCE.OneHot]),\n    [\"UINT32ARRAY\" /* WGPUENTRYTYPE.UINT32ARRAY */]: new Set([WEBGPUDISTANCE.HAMMING, WEBGPUDISTANCE.EUCLIDEAN, WEBGPUDISTANCE.MANHATTAN, WEBGPUDISTANCE.MONOMER_CHEMICAL_DISTANCE, WEBGPUDISTANCE.LEVENSTEIN, WEBGPUDISTANCE.NEEDLEMAN_WUNSCH, WEBGPUDISTANCE.TANIMOTO, WEBGPUDISTANCE.COSINE, WEBGPUDISTANCE.SOKAL, WEBGPUDISTANCE.ASYMMETRIC, WEBGPUDISTANCE.OneHot, WEBGPUDISTANCE.Difference]),\n    [\"INT32ARRAY\" /* WGPUENTRYTYPE.INT32ARRAY */]: new Set([WEBGPUDISTANCE.EUCLIDEAN, WEBGPUDISTANCE.MANHATTAN, WEBGPUDISTANCE.OneHot, WEBGPUDISTANCE.Difference]),\n    [\"FLOAT32ARRAY\" /* WGPUENTRYTYPE.FLOAT32ARRAY */]: new Set([WEBGPUDISTANCE.EUCLIDEAN, WEBGPUDISTANCE.MANHATTAN, WEBGPUDISTANCE.Difference]),\n    [\"NUMBER\" /* WGPUENTRYTYPE.NUMBER */]: new Set([WEBGPUDISTANCE.EUCLIDEAN, WEBGPUDISTANCE.MANHATTAN, WEBGPUDISTANCE.Difference]),\n    [\"BITARRAY\" /* WGPUENTRYTYPE.BITARRAY */]: new Set([WEBGPUDISTANCE.TANIMOTO, WEBGPUDISTANCE.COSINE, WEBGPUDISTANCE.SOKAL, WEBGPUDISTANCE.ASYMMETRIC])\n};\n//# sourceMappingURL=webGPU-multicol-distances.js.map","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nlet gpuAdapter = null;\nlet gpuDevice = null;\nexport function getGPUDevice() {\n    return __awaiter(this, void 0, void 0, function* () {\n        if (!gpuAdapter) {\n            //reason: only here we get the gpuAdapter\n            // eslint-disable-next-line no-restricted-syntax\n            gpuAdapter = yield navigator.gpu.requestAdapter({ powerPreference: 'high-performance' });\n            if (gpuAdapter == null)\n                return null;\n        }\n        let isLost = false;\n        if (gpuDevice) {\n            gpuDevice.lost.then(() => {\n                isLost = true;\n            });\n            yield new Promise((r) => setTimeout(r, 10)); // wait to see if the device is lost\n        }\n        if (!gpuDevice || isLost) {\n            const requiredBufferSize = 1000000000; // ~1000MB\n            const adapterLimits = gpuAdapter.limits;\n            const buffferSizeLimit = adapterLimits.maxBufferSize;\n            const storageBufferSizeLimit = adapterLimits.maxStorageBufferBindingSize;\n            try {\n                gpuDevice = yield gpuAdapter.requestDevice({ requiredLimits: {\n                        maxBufferSize: Math.min(buffferSizeLimit, requiredBufferSize),\n                        maxStorageBufferBindingSize: Math.min(storageBufferSizeLimit, requiredBufferSize)\n                    } });\n                return gpuDevice;\n            }\n            catch (e) {\n                console.error('Failed to create device with required limits', e);\n                gpuDevice = yield gpuAdapter.requestDevice();\n                return gpuDevice;\n            }\n        }\n        return gpuDevice;\n    });\n}\nexport function getGPUAdapterDescription() {\n    return __awaiter(this, void 0, void 0, function* () {\n        if (!gpuAdapter) {\n            // reason: only here we get the gpuAdapter\n            // eslint-disable-next-line no-restricted-syntax\n            gpuAdapter = yield navigator.gpu.requestAdapter();\n            if (gpuAdapter == null)\n                return null;\n        }\n        const info = yield gpuAdapter.requestAdapterInfo();\n        if (!info)\n            return null;\n        const outString = replaceEmptyString(info.description, replaceEmptyString(info.vendor, 'No GPU description available'));\n        return outString;\n    });\n}\nfunction replaceEmptyString(str, replacement) {\n    return !str || str == '' ? replacement : str;\n}\n//# sourceMappingURL=getGPUDevice.js.map","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport { WEBGSLAGGREGATION, WEBGSLAGGREGATIONFUNCTIONS, } from '../multi-col-distances/webGPU-aggregation';\nimport { WEBGPUDISTANCE, webGPUFunctions, } from '../multi-col-distances/webGPU-multicol-distances';\nimport { webGPUProcessInfo } from '../preprocessing/webGPU-process-info';\nimport { getGPUDevice } from '../getGPUDevice';\n/** generate sparse matrix based on list of lists of entries.\n *  these entries are each encoded as Uint32Array or FLOAT32Array (depending on their type).\n * for example, sequences would be encoded as Uint32Array based on char code of the letter at each position.\n * [65, 66, 67, 68, 69] would be a sequence of 5 letters.\n * for chemical fingerprints, it would be a binary array of 0s and 1s,\n * represented as Uint32Array(_data property of DG bitarray).\n *\n * Be ware that size of entryList, distanceMetrics, weights and options must be the same.\n * if there are no options for entries i, pass an empty object.\n * for now options are needed for\n * needleman-wunsch and monomer chemical distances: see {@link BioDistanceFnOptions} as for how it should be passed\n * numeric distances (Difference): {range: number} where range is the range of the values in the column (max - min).\n * in both cases, if options are not provided, they will be calculated automatically.\n */\nexport function multiColWebGPUSparseMatrix(entryList, // list of lists of entries, for multiple columns\nthreshold = 0.8, // similarity threshold, be ware that if you use too small threshold, there might be memory overflow...\ndistanceMetrics, // distance metrics for each column\naggregationFunction, // aggregation function for the distances\nweights, // weights for each column\noptions // supplementary options for each column\n) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const device = yield getGPUDevice();\n        if (!device)\n            return null; // if no device, return null, as we cannot do anything without it.\n        const availableDistanceMetrics = Object.values(WEBGPUDISTANCE);\n        if (distanceMetrics.some((metric) => !availableDistanceMetrics.includes(metric)))\n            throw new Error('Invalid distance metrics provided: ' + distanceMetrics.join(', '));\n        const availableAggregationFunctions = Object.values(WEBGSLAGGREGATION);\n        if (!availableAggregationFunctions.includes(aggregationFunction))\n            throw new Error('Invalid aggregation function provided: ' + aggregationFunction);\n        const maxDistance = 1 - threshold; // maximum distance\n        // first, check that all the supplementary options are provided and are the same length:\n        if (options.length !== entryList.length ||\n            options.length !== distanceMetrics.length ||\n            options.length !== weights.length) {\n            throw new Error('Options, weigths and distance functions must be provided for each column');\n        }\n        // check that all the entry lists are the same length\n        if (entryList.some((list) => list.length !== entryList[0].length))\n            throw new Error('All entry lists must be the same length');\n        const numOfColumns = entryList.length; // number of columns\n        const listSize = entryList[0].length; // size of each list (or column)\n        const processInfo = entryList.map((entry, i) => {\n            return webGPUProcessInfo(entry, distanceMetrics[i], i, options[i]);\n        });\n        if (numOfColumns === 0) {\n            throw new Error('No columns provided. Please provide at least one column of data.');\n        }\n        if (numOfColumns === 1)\n            aggregationFunction = WEBGSLAGGREGATION.MANHATTAN; // save a bit of time\n        // combine all struct types into one to put into the suppInfo struct.\n        let suppInfoWgsl = processInfo\n            .map((info) => info.suppInfoStructWgsl)\n            .filter((wgsl) => !!wgsl && wgsl != '')\n            .join(',\\n');\n        // structures in wgsl must have at least one member, so if we have no structures, we need to add a dummy one\n        let needsDummy = false;\n        if (!suppInfoWgsl || suppInfoWgsl.trim() == '') {\n            needsDummy = true;\n            suppInfoWgsl = '\\ndummy: f32\\n';\n        }\n        // combine all data wgsl struct code into one\n        const dataWgsl = processInfo.map((info) => info.dataStructWgsl).filter((wgsl) => !!wgsl && wgsl != '').join(',\\n');\n        // combine all array sizes into one array (easier for setting)\n        const arraySizes = new Uint32Array(numOfColumns * listSize);\n        processInfo.forEach((info, i) => {\n            arraySizes.set(info.arraySizes, i * listSize);\n        }); // array.flat is not as optimized as this\n        // if we try to map large arrays directly from GPU, sometimes, device disconnects. so we need to do it in chunks, a good number\n        // we found is 10000. So we will perform computations in chunks of 10000. meaning that we will dispatch 10000 threads at a time.\n        const numOfThreads = 10000;\n        // in this case we do not need to worry about complexity of the algorithm, as the 100 is low enaugh number, which is limited by memory usage.\n        const sparseResultSizePerThread = 100; // number of iterations per thread (number of pair comparisons)\n        const combinedComplexity = processInfo.reduce((a, b) => a + b.complexity, 0); // combined complexity of all the columns\n        const maxIterationsPerThread = Math.ceil(10000 / combinedComplexity); // maximum number of iterations per thread\n        const workGroupDivision = 10; // how many threads inside of one workgroup dimension (in this case 10 * 10 threads per workgroup)\n        const threadsPerWorkgroup = workGroupDivision * workGroupDivision;\n        const workgroupsDim = Math.ceil(Math.sqrt(Math.ceil(numOfThreads / threadsPerWorkgroup))); // how many workgroups per 2d dimension\n        const globalThreadDimSize = workgroupsDim * workGroupDivision; // how many threads per 2d dimension\n        const condensedDistanceMatrixSize = listSize * (listSize - 1) / 2; // size of the condensed distance matrix, this many comparisons will be made.\n        const dmChunkSizePerThread = Math.ceil(condensedDistanceMatrixSize / numOfThreads); // how many comparisons per thread\n        const module = device.createShaderModule({\n            label: 'Sparse matrix compute shader',\n            code: `\n    // each thread will perform ${sparseResultSizePerThread} iterations at one time, comparing ${sparseResultSizePerThread} pairs of entries.\n    // in total, each thread will perform at most ${dmChunkSizePerThread} comparisons.\n    // first is the result struct, containing is, js, and distances. each array with length of ${sparseResultSizePerThread},\n    // and also integer for how many pairs were found to be below threshold.\n    struct SparseResult {\n      i: array<array<u32, ${sparseResultSizePerThread}>, ${numOfThreads}>,\n      j: array<array<u32, ${sparseResultSizePerThread}>, ${numOfThreads}>,\n      distances: array<array<f32, ${sparseResultSizePerThread}>, ${numOfThreads}>,\n      found: array<u32, ${numOfThreads}>,\n      done: array<u32, ${numOfThreads}>\n    }\n    // struct for the data\n    struct ComputeInfo {\n      // start at cols and rows, and end at cols and rows for each thread, these will be calculated on cpu and passed to gpu.\n      startAtCols: array<u32, ${numOfThreads}>,\n      startAtRows: array<u32, ${numOfThreads}>,\n      endAtCols: array<u32, ${numOfThreads}>,\n      endAtRows: array<u32, ${numOfThreads}>,\n\n      // the ACTUALLY sizes of each entry\n      entrySizes: array<array<u32, ${listSize}>, ${numOfColumns}>,\n      // the weights for each entry\n      weights: array<f32, ${numOfColumns}>,\n      // the data for each entry\n      ${dataWgsl} // an example of the dataWgsl would be:\n      //data0: array<array<u32,20>,100>,\n      //data1: array<array<u32,20>,100>\n    }\n\n    // struct for the supplementary information\n    struct SuppInfo {\n      // struct containing all the supplementary info, like scoring matrix, alphabet indexes, range, etc.\n      ${suppInfoWgsl}\n    };\n\n    @group(0) @binding(0) var<storage, read_write> computeInfo: ComputeInfo;\n    @group(0) @binding(1) var<storage, read_write> suppInfo: SuppInfo;\n    @group(0) @binding(2) var<storage, read_write> results: SparseResult;\n    @compute @workgroup_size(${workGroupDivision}, ${workGroupDivision}) fn calcSparseMatrix(\n      @builtin(global_invocation_id) id: vec3<u32>\n    ) {\n      ${needsDummy ? `let otherDummy = suppInfo.dummy * 2;` : ''} // just to make sure that the suppInfo is not optimized out\n      let threadCol = id.x;\n      let threadRow = id.y;\n      let linearIndex = threadRow * ${globalThreadDimSize} + threadCol;\n      if (linearIndex >= ${numOfThreads}) {\n        return; // if we are out of bounds, return\n      }      \n      var startAtCol: u32 = computeInfo.startAtCols[linearIndex];\n      var startAtRow: u32 = computeInfo.startAtRows[linearIndex];\n      let endAtCol: u32 = min(computeInfo.endAtCols[linearIndex], ${listSize}u);\n      let endAtRow: u32 = min(computeInfo.endAtRows[linearIndex], ${listSize}u);\n      let is = &results.i[linearIndex];\n      let js = &results.j[linearIndex];\n      let distances = &results.distances[linearIndex];\n      results.found[linearIndex] = 0; // initialize the found counter\n      var found: u32 = 0;\n      if (results.done[linearIndex] > 0) {\n        return; // if we are done, return\n      }\n      for (var i = 0; i < ${maxIterationsPerThread}; i++) {\n        if (startAtCol >= endAtCol && startAtRow >= endAtRow) {\n          results.done[linearIndex] = 1;\n          break;\n        }\n        if (found >= ${sparseResultSizePerThread}) {\n          break;\n        }\n        let dist = combinedDistance(startAtCol, startAtRow);\n        if (dist <= ${maxDistance}) {\n          (*is)[found] = startAtCol;\n          (*js)[found] = startAtRow;\n          (*distances)[found] = dist;\n          found = found + 1;\n        }\n        startAtCol = startAtCol + 1;\n        if (startAtCol >= ${listSize}u) {\n          startAtRow += 1;\n          startAtCol = startAtRow + 1;\n        }\n      }\n      results.found[linearIndex] = found;\n      // update the startAtCols and startAtRows\n      computeInfo.startAtCols[linearIndex] = startAtCol;\n      computeInfo.startAtRows[linearIndex] = startAtRow;\n\n    }\n\n    // this will generate the distance script for each distance metric and then combine them into one\n    ${getCombinedDistanceScript(distanceMetrics, processInfo.map((info) => info.maxEntryLen), maxDistance, aggregationFunction)}\n\n\n    `\n        });\n        const pipeline = device.createComputePipeline({\n            label: 'sparse matrix compute pipeline',\n            layout: 'auto',\n            compute: {\n                module,\n                entryPoint: 'calcSparseMatrix',\n            },\n        });\n        // generate startAtCols, startAtRows, endAtCols, endAtRows\n        const startAtCols = new Uint32Array(numOfThreads);\n        const startAtRows = new Uint32Array(numOfThreads);\n        const endAtCols = new Uint32Array(numOfThreads);\n        const endAtRows = new Uint32Array(numOfThreads);\n        const chunkSize = Math.floor(condensedDistanceMatrixSize / numOfThreads); // size of the chunk per thread (in total)\n        let startRow = 0;\n        let startCol = 1;\n        console.time('GPUthreadStarts');\n        for (let i = 0; i < numOfThreads; i++) {\n            const endIdx = i === numOfThreads - 1 ? condensedDistanceMatrixSize - 1 : (i + 1) * chunkSize;\n            // fancy formulas to calculate the start and end indices for the condensed distance matrix for each thread start\n            const endRow = listSize - 2 - Math.floor(Math.sqrt(-8 * endIdx + 4 * listSize * (listSize - 1) - 7) / 2 - 0.5);\n            const endCol = endIdx - listSize * endRow + Math.floor((endRow + 1) * (endRow + 2) / 2);\n            startAtCols[i] = startCol;\n            startAtRows[i] = startRow;\n            endAtCols[i] = endCol;\n            endAtRows[i] = endRow;\n            startRow = endRow;\n            startCol = endCol;\n            // const startRow = values[0].length - 2 - Math.floor(\n            //   Math.sqrt(-8 * startIdx + 4 * values[0].length * (values[0].length - 1) - 7) / 2 - 0.5);\n            // const startCol = startIdx - values[0].length * startRow + Math.floor((startRow + 1) * (startRow + 2) / 2);\n        }\n        console.timeEnd('GPUthreadStarts');\n        // size of the computeInfo buffer\n        const computeInfoBuffer32Size = numOfThreads * 4 + // startAtCols, startAtRows, endAtCols, endAtRows\n            listSize * numOfColumns + // entrySizes\n            numOfColumns + // weights\n            processInfo.reduce((a, b) => a + b.sourceArraySize, 0);\n        // size of the suppInfo buffer\n        const suppInfoBuffer32Size = processInfo.reduce((a, b) => a + b.suppInfoSize, 0);\n        // size of the results buffer\n        const sparseMatrixEachArray32Size = sparseResultSizePerThread * numOfThreads;\n        const resultsBuffer32Size = 3 * sparseMatrixEachArray32Size + numOfThreads + numOfThreads; // i, j, distances, found, done\n        // create a buffer on the GPU to hold computeInfo\n        // beware that struct must be padded to 16 bytes, so we need to calculate the size of the struct in 32bit values\n        const computeInfoBufferSize = computeInfoBuffer32Size * Uint32Array.BYTES_PER_ELEMENT;\n        let paddedComputeInfoBufferSize = computeInfoBufferSize;\n        const remainder = computeInfoBufferSize & 15; // check if the size is a multiple of 16\n        if (remainder !== 0)\n            paddedComputeInfoBufferSize += 16 - remainder; // pad the size accordingly\n        const computeInfoBuffer = device.createBuffer({\n            label: 'compute info buffer',\n            size: paddedComputeInfoBufferSize,\n            usage: GPUBufferUsage.STORAGE |\n                GPUBufferUsage.COPY_SRC |\n                GPUBufferUsage.COPY_DST,\n            mappedAtCreation: true,\n        });\n        const mappedComputeInfoArrayBuffer = computeInfoBuffer.getMappedRange(); // get full buffer\n        // dynamic offset for the computeInfo buffer\n        let computeInfoOffSet = 0;\n        // first write the startAtCols, startAtRows, endAtCols, endAtRows\n        const startAtColsBufferView = new Uint32Array(mappedComputeInfoArrayBuffer, computeInfoOffSet, numOfThreads);\n        startAtColsBufferView.set(startAtCols);\n        computeInfoOffSet += numOfThreads * Uint32Array.BYTES_PER_ELEMENT; // array of 32bit values\n        const startAtRowsBufferView = new Uint32Array(mappedComputeInfoArrayBuffer, computeInfoOffSet, numOfThreads);\n        startAtRowsBufferView.set(startAtRows);\n        computeInfoOffSet += numOfThreads * Uint32Array.BYTES_PER_ELEMENT; // array of 32bit values\n        const endAtColsBufferView = new Uint32Array(mappedComputeInfoArrayBuffer, computeInfoOffSet, numOfThreads);\n        endAtColsBufferView.set(endAtCols);\n        computeInfoOffSet += numOfThreads * Uint32Array.BYTES_PER_ELEMENT; // array of 32bit values\n        const endAtRowsBufferView = new Uint32Array(mappedComputeInfoArrayBuffer, computeInfoOffSet, numOfThreads);\n        endAtRowsBufferView.set(endAtRows);\n        computeInfoOffSet += numOfThreads * Uint32Array.BYTES_PER_ELEMENT; // array of 32bit values\n        // then write the entrySizes\n        const entrySizesView = new Uint32Array(mappedComputeInfoArrayBuffer, computeInfoOffSet, arraySizes.length);\n        entrySizesView.set(arraySizes);\n        computeInfoOffSet += arraySizes.length * Uint32Array.BYTES_PER_ELEMENT; // array of 32bit values\n        // then write the weights\n        const weightsView = new Float32Array(mappedComputeInfoArrayBuffer, computeInfoOffSet, numOfColumns);\n        weightsView.set(weights);\n        computeInfoOffSet += numOfColumns * Float32Array.BYTES_PER_ELEMENT;\n        // finally, write the data itself\n        for (const info of processInfo) {\n            const ArrayConstructor = info.EncodedArrayConstructor;\n            const chunkSize = info.sourceArraySize;\n            const dataView = new ArrayConstructor(mappedComputeInfoArrayBuffer, computeInfoOffSet, chunkSize); //new ArrayConstructor(computeInfoBuffer.getMappedRange(computeInfoOffSet, chunkByteSize));\n            dataView.set(info.flatSourceArray);\n            computeInfoOffSet += chunkSize * ArrayConstructor.BYTES_PER_ELEMENT;\n        }\n        // we are done at this point.\n        computeInfoBuffer.unmap();\n        // create a buffer on the GPU to hold suppInfo\n        // same here, we need to pad the size of the struct to 16 bytes\n        const suppInfoBufferSize = suppInfoBuffer32Size * Uint32Array.BYTES_PER_ELEMENT;\n        let paddedSuppInfoBufferSize = suppInfoBufferSize;\n        const suppInfoRemainder = suppInfoBufferSize & 15; // check if the size is a multiple of 16\n        if (suppInfoRemainder !== 0)\n            paddedSuppInfoBufferSize += 16 - suppInfoRemainder; // pad the size accordingly\n        paddedSuppInfoBufferSize = Math.max(paddedSuppInfoBufferSize, 16);\n        const suppInfoBuffer = device.createBuffer({\n            label: 'supp info buffer',\n            size: paddedSuppInfoBufferSize,\n            usage: GPUBufferUsage.STORAGE |\n                GPUBufferUsage.COPY_SRC |\n                GPUBufferUsage.COPY_DST,\n            mappedAtCreation: true,\n        });\n        const mappedSuppInfoArrayBuffer = suppInfoBuffer.getMappedRange(); // get full buffer\n        let suppInfoOffSet = 0;\n        for (const info of processInfo) {\n            if (info.suppInfoBuffer && info.suppInfoBuffer.byteLength > 0 && info.suppInfoSize > 0) {\n                const ArrayConstructor = info.suppInfoType === \"UINT32ARRAY\" /* WGPUENTRYTYPE.UINT32ARRAY */ ? Uint32Array : Float32Array;\n                const suppInfoView = new ArrayConstructor(mappedSuppInfoArrayBuffer, suppInfoOffSet, info.suppInfoBuffer.length); //new ArrayConstructor(suppInfoBuffer.getMappedRange(suppInfoOffSet, info.suppInfoBuffer.byteLength));\n                suppInfoView.set(info.suppInfoBuffer);\n                suppInfoOffSet += info.suppInfoBuffer.byteLength; // info.suppInfoBuffer.length * ArrayConstructor.BYTES_PER_ELEMENT;\n            }\n        }\n        if (suppInfoOffSet === 0) {\n            const dummyView = new Uint32Array(mappedSuppInfoArrayBuffer, 0, 4); //new Uint32Array(suppInfoBuffer.getMappedRange(0, 16));\n            dummyView.set([1, 1, 1, 1]);\n        }\n        suppInfoBuffer.unmap();\n        // create a buffer for the results\n        const resultsBufferSize = resultsBuffer32Size * Uint32Array.BYTES_PER_ELEMENT;\n        let paddedResultsBufferSize = resultsBufferSize;\n        const resultsRemainder = resultsBufferSize & 15; // check if the size is a multiple of 16\n        if (resultsRemainder !== 0)\n            paddedResultsBufferSize += 16 - resultsRemainder; // pad the size accordingly\n        const resultsBuffer = device.createBuffer({\n            label: 'results buffer',\n            size: paddedResultsBufferSize,\n            usage: GPUBufferUsage.STORAGE |\n                GPUBufferUsage.COPY_SRC\n        });\n        // Setup a bindGroup to tell the shader which\n        // buffer to use for the computation\n        const bindGroup = device.createBindGroup({\n            label: 'bindGroup for sparse matrix buffer',\n            layout: pipeline.getBindGroupLayout(0),\n            entries: [\n                { binding: 0, resource: { buffer: computeInfoBuffer } },\n                { binding: 1, resource: { buffer: suppInfoBuffer } },\n                { binding: 2, resource: { buffer: resultsBuffer } },\n            ],\n        });\n        //const pairComparisonsPerPass = maxIterationsPerThread * numOfThreads;\n        //const passes = Math.ceil(condensedDistanceMatrixSize / pairComparisonsPerPass);\n        // we will distpatch this many passes to the GPU, and it will handle indexes all by itself.\n        // we already copied the start/end information to it, so it will know where to start and end on each pass.\n        const resultsOutBuffer = device.createBuffer({\n            label: 'results out buffer',\n            size: resultsBuffer.size,\n            usage: GPUBufferUsage.MAP_READ | GPUBufferUsage.COPY_DST,\n        });\n        const resultIs = [];\n        const resultJs = [];\n        const resultDistances = [];\n        //let combinedFound = 0;\n        let isAllDone = false;\n        while (!isAllDone) {\n            // Encode commands to do the computation\n            const encoder = device.createCommandEncoder({\n                label: 'distance encoder',\n            });\n            const pass = encoder.beginComputePass({\n                label: 'distance compute pass',\n            });\n            pass.setPipeline(pipeline);\n            pass.setBindGroup(0, bindGroup);\n            pass.dispatchWorkgroups(workgroupsDim, workgroupsDim);\n            pass.end();\n            encoder.copyBufferToBuffer(resultsBuffer, 0, resultsOutBuffer, 0, resultsOutBuffer.size);\n            // Finish encoding and submit the commands\n            const commandBuffer = encoder.finish();\n            device.queue.submit([commandBuffer]);\n            // Read the results\n            yield device.queue.onSubmittedWorkDone();\n            yield resultsOutBuffer.mapAsync(GPUMapMode.READ);\n            const resultsOutArrayBuffer = resultsOutBuffer.getMappedRange();\n            // read the results\n            let resultOffset = 0;\n            const resultsI = new Uint32Array(resultsOutArrayBuffer, resultOffset, sparseMatrixEachArray32Size);\n            resultOffset += sparseMatrixEachArray32Size * Uint32Array.BYTES_PER_ELEMENT;\n            const resultsJ = new Uint32Array(resultsOutArrayBuffer, resultOffset, sparseMatrixEachArray32Size);\n            resultOffset += sparseMatrixEachArray32Size * Uint32Array.BYTES_PER_ELEMENT;\n            const resultsDistances = new Float32Array(resultsOutArrayBuffer, resultOffset, sparseMatrixEachArray32Size);\n            resultOffset += sparseMatrixEachArray32Size * Float32Array.BYTES_PER_ELEMENT;\n            const resultsFound = new Uint32Array(resultsOutArrayBuffer, resultOffset, numOfThreads);\n            resultOffset += numOfThreads * Uint32Array.BYTES_PER_ELEMENT;\n            const resultsDone = new Uint32Array(resultsOutArrayBuffer, resultOffset, numOfThreads);\n            isAllDone = resultsDone.every((d) => d === 1);\n            const totalResults = resultsFound.reduce((a, b) => a + b, 0);\n            const combinedI = new Uint32Array(totalResults);\n            const combinedJ = new Uint32Array(totalResults);\n            const combinedDistances = new Float32Array(totalResults);\n            let combinedOffset = 0;\n            for (let resI = 0; resI < resultsFound.length; resI++) {\n                const found = resultsFound[resI];\n                if (found === 0)\n                    continue;\n                combinedI.set(resultsI.subarray(resI * sparseResultSizePerThread, resI * sparseResultSizePerThread + found), combinedOffset);\n                combinedJ.set(resultsJ.subarray(resI * sparseResultSizePerThread, resI * sparseResultSizePerThread + found), combinedOffset);\n                combinedDistances.set(resultsDistances.subarray(resI * sparseResultSizePerThread, resI * sparseResultSizePerThread + found), combinedOffset);\n                combinedOffset += found;\n            }\n            resultIs.push(combinedI);\n            resultJs.push(combinedJ);\n            resultDistances.push(combinedDistances);\n            resultsOutBuffer.unmap();\n        }\n        const totalSize = resultIs.reduce((a, b) => a + b.length, 0);\n        const finalI = new Uint32Array(totalSize);\n        const finalJ = new Uint32Array(totalSize);\n        const finalDistances = new Float32Array(totalSize);\n        let finalOffset = 0;\n        for (let i = 0; i < resultIs.length; i++) {\n            finalI.set(resultIs[i], finalOffset);\n            finalJ.set(resultJs[i], finalOffset);\n            finalDistances.set(resultDistances[i], finalOffset);\n            finalOffset += resultIs[i].length;\n        }\n        // as rule mandates, destroy all buffers.\n        computeInfoBuffer.destroy();\n        suppInfoBuffer.destroy();\n        resultsBuffer.destroy();\n        resultsOutBuffer.destroy();\n        return { i: finalI, j: finalJ, distance: finalDistances };\n    });\n}\nfunction getCombinedDistanceScript(distanceMetrics, maxEntryLens, maxDistance, aggregation) {\n    const distanceWgsls = distanceMetrics.map((metric, i) => {\n        return `\n        fn distanceScript${i}(aIndex: u32, bIndex: u32) -> f32 {\n          let a = computeInfo.data${i}[aIndex];\n          let b = computeInfo.data${i}[bIndex];\n          let maxDistance: f32 = ${maxDistance};\n          ${webGPUFunctions[metric](maxEntryLens[i], i)}\n        }\n      `;\n    });\n    const allDistanceScripts = distanceWgsls.join('\\n');\n    const combineDistancesScript = `\n      fn combinedDistance(aIndex: u32, bIndex: u32) -> f32 {\n        var distances: array<f32, ${distanceMetrics.length}>;\n        ${distanceMetrics.map((_, i) => `distances[${i}] = distanceScript${i}(aIndex, bIndex);`).join('\\n')}\n        ${WEBGSLAGGREGATIONFUNCTIONS[aggregation](distanceMetrics.length)}\n      }\n    \n    `;\n    return allDistanceScripts + '\\n' + combineDistancesScript;\n}\n//# sourceMappingURL=webGPU-sparse-matrix.js.map","export var DistanceAggregationMethods;\n(function (DistanceAggregationMethods) {\n    DistanceAggregationMethods[\"EUCLIDEAN\"] = \"EUCLIDEAN\";\n    DistanceAggregationMethods[\"MANHATTAN\"] = \"MANHATTAN\";\n})(DistanceAggregationMethods || (DistanceAggregationMethods = {}));\n;\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJ0eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLENBQU4sSUFBWSwwQkFHVDtBQUhILFdBQVksMEJBQTBCO0lBQ2xDLHFEQUF1QixDQUFBO0lBQ3ZCLHFEQUF1QixDQUFBO0FBQ3pCLENBQUMsRUFIUywwQkFBMEIsS0FBMUIsMEJBQTBCLFFBR25DO0FBQUEsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBlbnVtIERpc3RhbmNlQWdncmVnYXRpb25NZXRob2RzIHtcbiAgICBFVUNMSURFQU4gPSAnRVVDTElERUFOJyxcbiAgICBNQU5IQVRUQU4gPSAnTUFOSEFUVEFOJyxcbiAgfTtcblxuZXhwb3J0IHR5cGUgRGlzdGFuY2VBZ2dyZWdhdGlvbk1ldGhvZCA9IGtleW9mIHR5cGVvZiBEaXN0YW5jZUFnZ3JlZ2F0aW9uTWV0aG9kcztcbiJdfQ==","/* eslint-disable max-len */\nimport { distanceFunctionComplexity, TypeSupportedDistances, WEBGPUDISTANCE } from '../multi-col-distances/webGPU-multicol-distances';\nexport function webGPUProcessInfo(entryList, distanceMetric = WEBGPUDISTANCE.HAMMING, entryIndex, // index of the entries in the list of lists that we want to process\noptions = { gapOpenPenalty: 1.0, gapExtensionPenalty: 0.6 }) {\n    var _a, _b;\n    let entryType = null;\n    const encodedList = (() => {\n        if (entryList.some((e) => typeof e === 'string')) {\n            entryType = \"STRING\" /* WGPUENTRYTYPE.STRING */;\n            return entryList.map((entry) => new Uint32Array(entry.split('').map((c) => c.charCodeAt(0))));\n        }\n        if (entryList.some((e) => typeof e === 'number')) {\n            entryType = \"NUMBER\" /* WGPUENTRYTYPE.NUMBER */;\n            return entryList.map((entry) => new Float32Array([entry]));\n        }\n        if (typeof entryList[0] == 'object' && entryList.some((e) => '_data' in e && '_length' in e)) {\n            entryType = \"BITARRAY\" /* WGPUENTRYTYPE.BITARRAY */;\n            return entryList.map((entry) => entry._data);\n        }\n        if (entryList.some((e) => e instanceof Float32Array)) {\n            entryType = \"FLOAT32ARRAY\" /* WGPUENTRYTYPE.FLOAT32ARRAY */;\n            return entryList;\n        }\n        if (entryList.some((e) => e instanceof Uint32Array)) {\n            entryType = \"UINT32ARRAY\" /* WGPUENTRYTYPE.UINT32ARRAY */;\n            return entryList;\n        }\n        if (entryList.some((e) => e instanceof Int32Array)) {\n            entryType = \"INT32ARRAY\" /* WGPUENTRYTYPE.INT32ARRAY */;\n            return entryList;\n        }\n        //return entryList as Uint32Array[];\n    })();\n    if (!encodedList || !entryType)\n        throw new Error('Invalid entry type, could not determine entry type from input list');\n    const encodedListType = encodedList[0] instanceof Int32Array ? \"INT32ARRAY\" /* WGPUENTRYTYPE.INT32ARRAY */ :\n        encodedList[0] instanceof Float32Array ? \"FLOAT32ARRAY\" /* WGPUENTRYTYPE.FLOAT32ARRAY */ : \"UINT32ARRAY\" /* WGPUENTRYTYPE.UINT32ARRAY */;\n    // sizes of each entries might differ, so we need to keep track of that for some distance metrics, like hamming for example\n    const arraySizes = new Uint32Array(encodedList.map((arr) => arr.length));\n    if (!TypeSupportedDistances[entryType] || !TypeSupportedDistances[entryType].has(distanceMetric))\n        throw new Error(`Distance metric '${distanceMetric}' not supported for entry type '${entryType}'`);\n    const maxEntryLen = arraySizes.reduce((a, b) => Math.max(a, b), 0);\n    // get the complexity of used algorithm\n    const complexity = distanceFunctionComplexity[distanceMetric](maxEntryLen);\n    const EncodedArrayConstructor = encodedListType === \"INT32ARRAY\" /* WGPUENTRYTYPE.INT32ARRAY */ ? Int32Array :\n        (encodedListType === \"FLOAT32ARRAY\" /* WGPUENTRYTYPE.FLOAT32ARRAY */ ? Float32Array : Uint32Array);\n    const flatSourceArray = new EncodedArrayConstructor(encodedList.length * maxEntryLen);\n    // when setting, we need to set each array at a specific offset, which is controlled by maxArrayLen because each array might have different sizes.\n    // this way we will get correct matrix representation in the compute shader\n    encodedList.forEach((seq, i) => {\n        flatSourceArray.set(seq, i * maxEntryLen);\n    });\n    // NB! all this before the line was generic, now we need to calculate some specific things for some specific distance metrics\n    // initialize supp info line that will be included in the final shader;\n    let suppInfoStructWgsl = ''; // the code that will be included in the struct of suppInfo\n    let suppInfoSize = 0;\n    let suppInfoType = \"FLOAT32ARRAY\" /* WGPUENTRYTYPE.FLOAT32ARRAY */;\n    let suppInfoBuffer = null;\n    if (distanceMetric === WEBGPUDISTANCE.NEEDLEMAN_WUNSCH || distanceMetric === WEBGPUDISTANCE.MONOMER_CHEMICAL_DISTANCE) {\n        let maxMonomerIndex = options.scoringMatrix && options.alphabetIndexes ?\n            Object.keys(options.alphabetIndexes).reduce((prev, n) => Math.max(prev, n.charCodeAt(0)), 0) : -1;\n        // generate default similarity matrix if it is not provided\n        if (!options.alphabetIndexes || !options.scoringMatrix) {\n            for (let i = 0; i < flatSourceArray.length; i++) {\n                if (flatSourceArray[i] > maxMonomerIndex)\n                    maxMonomerIndex = flatSourceArray[i];\n            }\n            options.scoringMatrix =\n                new Array(maxMonomerIndex + 1).fill(null).map(() => new Array(maxMonomerIndex + 1).fill(0));\n            options.alphabetIndexes = {};\n            for (let i = 0; i < options.scoringMatrix.length; i++) {\n                options.scoringMatrix[i][i] = 1;\n                options.alphabetIndexes[String.fromCharCode(i)] = i;\n            }\n        }\n        const similarityMatrixSize = (maxMonomerIndex + 1) * (maxMonomerIndex + 1);\n        const transferedSimilarityMatrix = new Array(maxMonomerIndex + 1).fill(null).map(() => new Float32Array(maxMonomerIndex + 1));\n        // set diagonal to 1\n        for (let i = 0; i < maxMonomerIndex + 1; i++)\n            transferedSimilarityMatrix[i][i] = 1;\n        const alphabetIndexes = options.alphabetIndexes;\n        for (const key of Object.keys(alphabetIndexes)) {\n            for (const key2 of Object.keys(alphabetIndexes)) {\n                if (key === key2)\n                    continue;\n                transferedSimilarityMatrix[key.charCodeAt(0)][key2.charCodeAt(0)] =\n                    options.scoringMatrix[alphabetIndexes[key]][alphabetIndexes[key2]];\n            }\n        }\n        // in memory layout, we will have 2 float32 s for gapOpen and gapExtension penalties, and then f32 array<array<f32>> for similarity matrix.\n        // because of primitives, there will be no padding, so we can calculate the size directly\n        suppInfoSize = 2 + similarityMatrixSize;\n        suppInfoType = \"FLOAT32ARRAY\" /* WGPUENTRYTYPE.FLOAT32ARRAY */;\n        suppInfoBuffer = new Float32Array(suppInfoSize);\n        suppInfoBuffer[0] = (_a = options.gapOpenPenalty) !== null && _a !== void 0 ? _a : 1.0;\n        suppInfoBuffer[1] = (_b = options.gapExtensionPenalty) !== null && _b !== void 0 ? _b : 0.6;\n        let offset = 2;\n        for (let i = 0; i < transferedSimilarityMatrix.length; i++) {\n            suppInfoBuffer.set(transferedSimilarityMatrix[i], offset);\n            offset += transferedSimilarityMatrix[i].length;\n        }\n        suppInfoStructWgsl = `\n            gapOpenPenalty${entryIndex}: f32,\n            gapExtensionPenalty${entryIndex}: f32,\n            similarityMatrix${entryIndex}: array<array<f32, ${maxMonomerIndex + 1}>, ${maxMonomerIndex + 1}>`;\n    }\n    else if (distanceMetric === WEBGPUDISTANCE.Difference) {\n        // for difference, we need range of values for normalization of the difference\n        if (!options.range || typeof options.range !== 'number' || options.range <= 0) {\n            const min = flatSourceArray.reduce((a, b) => Math.min(a, b), flatSourceArray[0]);\n            const max = flatSourceArray.reduce((a, b) => Math.max(a, b), flatSourceArray[0]);\n            options.range = max - min;\n        }\n        if (options.range <= 0)\n            options.range = 1.0; // this means that all values are the same, and all distances will produce 0.\n        suppInfoSize = 1;\n        suppInfoType = \"FLOAT32ARRAY\" /* WGPUENTRYTYPE.FLOAT32ARRAY */;\n        suppInfoBuffer = new Float32Array([options.range]);\n        suppInfoStructWgsl = `\n            range${entryIndex}: f32`;\n    }\n    const dataTypeWGSL = flatSourceArray instanceof Int32Array ? 'i32' : (flatSourceArray instanceof Float32Array ? 'f32' : 'u32');\n    const dataStructWgsl = `data${entryIndex}: array<array<${dataTypeWGSL}, ${maxEntryLen}>, ${encodedList.length}>`;\n    // for now, other distances do not require any additional information, so we can skip that\n    return {\n        flatSourceArray,\n        sourceArraySize: flatSourceArray.length,\n        maxEntryLen,\n        arraySizes,\n        complexity,\n        suppInfoBuffer,\n        suppInfoSize,\n        suppInfoType: suppInfoType,\n        suppInfoStructWgsl,\n        entryType,\n        dataTypeWGSL,\n        dataStructWgsl,\n        EncodedArrayConstructor\n    };\n}\n//# sourceMappingURL=webGPU-process-info.js.map","import { DistanceAggregationMethods } from './types';\nexport const isNil = (x) => x === null || x === undefined;\nexport function insertSmaller(distancesAr, indexes, num, index) {\n    if (num > distancesAr[distancesAr.length - 1])\n        return;\n    const newPosition = distancesAr.findIndex((v) => num < v);\n    distancesAr.pop();\n    distancesAr.splice(newPosition, 0, num);\n    indexes.pop();\n    indexes.splice(newPosition, 0, index);\n}\nexport function insertLarger(distancesAr, indexes, num, index) {\n    if (num < distancesAr[distancesAr.length - 1])\n        return;\n    const newPosition = distancesAr.findIndex((v) => num > v);\n    distancesAr.pop();\n    distancesAr.splice(newPosition, 0, num);\n    indexes.pop();\n    indexes.splice(newPosition, 0, index);\n}\nexport function getAggregationFunction(aggregationMethod, weights) {\n    switch (aggregationMethod) {\n        case DistanceAggregationMethods.MANHATTAN:\n            return (vs) => vs.reduce((acc, val, idx) => acc + val * weights[idx], 0);\n        default:\n            return (vs) => {\n                // euclidean\n                const sum = vs.reduce((acc, val, idx) => acc + (val * weights[idx]) ** 2, 0);\n                return Math.sqrt(sum);\n            };\n    }\n}\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJ1dGlscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQTRCLDBCQUEwQixFQUFDLE1BQU0sU0FBUyxDQUFDO0FBRTlFLE1BQU0sQ0FBQyxNQUFNLEtBQUssR0FBRyxDQUFDLENBQU0sRUFBRSxFQUFFLENBQUMsQ0FBQyxLQUFLLElBQUksSUFBSSxDQUFDLEtBQUssU0FBUyxDQUFDO0FBRS9ELE1BQU0sVUFBVSxhQUFhLENBQUMsV0FBcUIsRUFBRSxPQUFpQixFQUFFLEdBQVcsRUFBRSxLQUFhO0lBQ2hHLElBQUksR0FBRyxHQUFHLFdBQVcsQ0FBQyxXQUFXLENBQUMsTUFBTSxHQUFDLENBQUMsQ0FBQztRQUN6QyxPQUFPO0lBRVQsTUFBTSxXQUFXLEdBQUcsV0FBVyxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsR0FBRyxHQUFHLENBQUMsQ0FBQyxDQUFDO0lBQzFELFdBQVcsQ0FBQyxHQUFHLEVBQUUsQ0FBQztJQUNsQixXQUFXLENBQUMsTUFBTSxDQUFDLFdBQVcsRUFBRSxDQUFDLEVBQUUsR0FBRyxDQUFDLENBQUM7SUFDeEMsT0FBTyxDQUFDLEdBQUcsRUFBRSxDQUFDO0lBQ2QsT0FBTyxDQUFDLE1BQU0sQ0FBQyxXQUFXLEVBQUUsQ0FBQyxFQUFFLEtBQUssQ0FBQyxDQUFDO0FBQ3hDLENBQUM7QUFFRCxNQUFNLFVBQVUsWUFBWSxDQUFDLFdBQXFCLEVBQUUsT0FBaUIsRUFBRSxHQUFXLEVBQUUsS0FBYTtJQUMvRixJQUFJLEdBQUcsR0FBRyxXQUFXLENBQUMsV0FBVyxDQUFDLE1BQU0sR0FBQyxDQUFDLENBQUM7UUFDekMsT0FBTztJQUVULE1BQU0sV0FBVyxHQUFHLFdBQVcsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLEdBQUcsR0FBRyxDQUFDLENBQUMsQ0FBQztJQUMxRCxXQUFXLENBQUMsR0FBRyxFQUFFLENBQUM7SUFDbEIsV0FBVyxDQUFDLE1BQU0sQ0FBQyxXQUFXLEVBQUUsQ0FBQyxFQUFFLEdBQUcsQ0FBQyxDQUFDO0lBQ3hDLE9BQU8sQ0FBQyxHQUFHLEVBQUUsQ0FBQztJQUNkLE9BQU8sQ0FBQyxNQUFNLENBQUMsV0FBVyxFQUFFLENBQUMsRUFBRSxLQUFLLENBQUMsQ0FBQztBQUN4QyxDQUFDO0FBRUQsTUFBTSxVQUFVLHNCQUFzQixDQUNwQyxpQkFBNEMsRUFBRSxPQUFpQjtJQUUvRCxRQUFRLGlCQUFpQixFQUFFLENBQUM7UUFDMUIsS0FBSywwQkFBMEIsQ0FBQyxTQUFTO1lBQ3ZDLE9BQU8sQ0FBQyxFQUFZLEVBQUUsRUFBRSxDQUFDLEVBQUUsQ0FBQyxNQUFNLENBQUMsQ0FBQyxHQUFHLEVBQUUsR0FBRyxFQUFFLEdBQUcsRUFBRSxFQUFFLENBQUMsR0FBRyxHQUFHLEdBQUcsR0FBRyxPQUFPLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFDckY7WUFDRSxPQUFPLENBQUMsRUFBWSxFQUFFLEVBQUU7Z0JBQ3RCLFlBQVk7Z0JBQ1osTUFBTSxHQUFHLEdBQUcsRUFBRSxDQUFDLE1BQU0sQ0FBQyxDQUFDLEdBQUcsRUFBRSxHQUFHLEVBQUUsR0FBRyxFQUFFLEVBQUUsQ0FBQyxHQUFHLEdBQUcsQ0FBQyxHQUFHLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQyxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDO2dCQUM3RSxPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7WUFDeEIsQ0FBQyxDQUFDO0lBQ04sQ0FBQztBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0Rpc3RhbmNlQWdncmVnYXRpb25NZXRob2QsIERpc3RhbmNlQWdncmVnYXRpb25NZXRob2RzfSBmcm9tICcuL3R5cGVzJztcblxuZXhwb3J0IGNvbnN0IGlzTmlsID0gKHg6IGFueSkgPT4geCA9PT0gbnVsbCB8fCB4ID09PSB1bmRlZmluZWQ7XG5cbmV4cG9ydCBmdW5jdGlvbiBpbnNlcnRTbWFsbGVyKGRpc3RhbmNlc0FyOiBudW1iZXJbXSwgaW5kZXhlczogbnVtYmVyW10sIG51bTogbnVtYmVyLCBpbmRleDogbnVtYmVyKSB7XG4gIGlmIChudW0gPiBkaXN0YW5jZXNBcltkaXN0YW5jZXNBci5sZW5ndGgtMV0pXG4gICAgcmV0dXJuO1xuXG4gIGNvbnN0IG5ld1Bvc2l0aW9uID0gZGlzdGFuY2VzQXIuZmluZEluZGV4KCh2KSA9PiBudW0gPCB2KTtcbiAgZGlzdGFuY2VzQXIucG9wKCk7XG4gIGRpc3RhbmNlc0FyLnNwbGljZShuZXdQb3NpdGlvbiwgMCwgbnVtKTtcbiAgaW5kZXhlcy5wb3AoKTtcbiAgaW5kZXhlcy5zcGxpY2UobmV3UG9zaXRpb24sIDAsIGluZGV4KTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGluc2VydExhcmdlcihkaXN0YW5jZXNBcjogbnVtYmVyW10sIGluZGV4ZXM6IG51bWJlcltdLCBudW06IG51bWJlciwgaW5kZXg6IG51bWJlcikge1xuICBpZiAobnVtIDwgZGlzdGFuY2VzQXJbZGlzdGFuY2VzQXIubGVuZ3RoLTFdKVxuICAgIHJldHVybjtcblxuICBjb25zdCBuZXdQb3NpdGlvbiA9IGRpc3RhbmNlc0FyLmZpbmRJbmRleCgodikgPT4gbnVtID4gdik7XG4gIGRpc3RhbmNlc0FyLnBvcCgpO1xuICBkaXN0YW5jZXNBci5zcGxpY2UobmV3UG9zaXRpb24sIDAsIG51bSk7XG4gIGluZGV4ZXMucG9wKCk7XG4gIGluZGV4ZXMuc3BsaWNlKG5ld1Bvc2l0aW9uLCAwLCBpbmRleCk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRBZ2dyZWdhdGlvbkZ1bmN0aW9uKFxuICBhZ2dyZWdhdGlvbk1ldGhvZDogRGlzdGFuY2VBZ2dyZWdhdGlvbk1ldGhvZCwgd2VpZ2h0czogbnVtYmVyW11cbik6ICh2YWx1ZXM6IG51bWJlcltdKSA9PiBudW1iZXIge1xuICBzd2l0Y2ggKGFnZ3JlZ2F0aW9uTWV0aG9kKSB7XG4gICAgY2FzZSBEaXN0YW5jZUFnZ3JlZ2F0aW9uTWV0aG9kcy5NQU5IQVRUQU46XG4gICAgICByZXR1cm4gKHZzOiBudW1iZXJbXSkgPT4gdnMucmVkdWNlKChhY2MsIHZhbCwgaWR4KSA9PiBhY2MgKyB2YWwgKiB3ZWlnaHRzW2lkeF0sIDApO1xuICAgIGRlZmF1bHQ6XG4gICAgICByZXR1cm4gKHZzOiBudW1iZXJbXSkgPT4ge1xuICAgICAgICAvLyBldWNsaWRlYW5cbiAgICAgICAgY29uc3Qgc3VtID0gdnMucmVkdWNlKChhY2MsIHZhbCwgaWR4KSA9PiBhY2MgKyAodmFsICogd2VpZ2h0c1tpZHhdKSAqKiAyLCAwKTtcbiAgICAgICAgcmV0dXJuIE1hdGguc3FydChzdW0pO1xuICAgICAgfTtcbiAgfVxufVxuIl19","import { DistanceAggregationMethods } from './types';\nimport { insertSmaller, isNil } from './utils';\nexport class SparseMatrixService {\n    constructor() {\n        this._workerCount = Math.max(navigator.hardwareConcurrency - 2, 1);\n    }\n    async calcMultiColumn(values, fnNames, threshold, opts = [{}], weights = [1], aggregationMethod = DistanceAggregationMethods.EUCLIDEAN) {\n        const matSize = values[0].length * (values[0].length - 1) / 2;\n        const chunkSize = Math.floor(matSize / this._workerCount);\n        const minThreshold = values[0].length > 20000 ?\n            await this.getMinimalThreshold(values, fnNames, opts, weights, aggregationMethod) : 0;\n        if (threshold < minThreshold) {\n            console.log(`using threshold ${minThreshold}`);\n            threshold = minThreshold;\n        }\n        opts.forEach((_, i) => opts[i]['threshold'] = threshold);\n        const promises = new Array(this._workerCount);\n        const workers = new Array(this._workerCount)\n            .fill(null).map(() => new Worker(new URL('./sparse-matrix-worker', import.meta.url)));\n        for (let idx = 0; idx < this._workerCount; idx++) {\n            promises[idx] = new Promise((resolveWorker, rejectWorker) => {\n                const startIdx = idx * chunkSize;\n                const endIdx = idx === this._workerCount - 1 ? matSize : (idx + 1) * chunkSize;\n                if (endIdx <= startIdx)\n                    resolveWorker({ i: new Int32Array(0), j: new Int32Array(0), distance: new Float32Array(0), idx });\n                workers[idx].postMessage({ values, startIdx, endIdx, threshold, fnNames, opts, weights, aggregationMethod });\n                workers[idx].onmessage = ({ data: { error, i, j, distance } }) => {\n                    if (error) {\n                        workers[idx].terminate();\n                        rejectWorker(error);\n                    }\n                    else {\n                        workers[idx].terminate();\n                        resolveWorker({ i, j, distance, idx });\n                    }\n                };\n            });\n        }\n        const results = await Promise.all(promises);\n        const fullSize = results.reduce((acc, val) => acc + val.i.length, 0);\n        const i = new Int32Array(fullSize);\n        const j = new Int32Array(fullSize);\n        const distance = new Float32Array(fullSize);\n        let offset = 0;\n        // setting the results\n        for (const res of results) {\n            i.set(res.i, offset);\n            j.set(res.j, offset);\n            distance.set(res.distance, offset);\n            offset += res.i.length;\n        }\n        return { i, j, distance };\n    }\n    async calc(values, fnName, threshold, opts = {}) {\n        //size of full matrix\n        return await this.calcMultiColumn([values], [fnName], threshold, [opts], [1]);\n    }\n    async getKNN(values, fnName, nNeighbours = 15, opts = {}) {\n        return await this.multiColumnKNN([values], [fnName], nNeighbours, [opts], [1]);\n    }\n    async getThresholdKNN(values, fnName, threshold = 0.8, opts = {}) {\n        return await this.multiColumnThresholdKnn([values], [fnName], threshold, [opts], [1]);\n    }\n    async multiColumnThresholdKnn(values, fnNames, threshold = 0.8, opts, weights, aggregationMethod = DistanceAggregationMethods.EUCLIDEAN) {\n        if (values.length !== fnNames.length || values.length !== opts.length || values.length !== weights.length)\n            throw new Error('values, distance functions, options and weights arrays should have the same length');\n        if (values.some((v) => v.length !== values[0].length))\n            throw new Error('all values arrays should have the same length');\n        const matSize = values[0].length * (values[0].length - 1) / 2;\n        const chunkSize = Math.floor(matSize / this._workerCount);\n        const promises = new Array(this._workerCount);\n        const workers = new Array(this._workerCount)\n            .fill(null).map(() => new Worker(new URL('./knn-threshold-worker', import.meta.url)));\n        for (let idx = 0; idx < this._workerCount; idx++) {\n            promises[idx] = new Promise((resolveWorker, rejectWorker) => {\n                const startIdx = idx * chunkSize;\n                const endIdx = idx === this._workerCount - 1 ? matSize : (idx + 1) * chunkSize;\n                if (endIdx <= startIdx)\n                    resolveWorker({ knnDistances: new Array(0), knnIndexes: new Array(0) });\n                workers[idx].postMessage({ values, startIdx, endIdx, fnNames, opts, threshold, weights, aggregationMethod });\n                workers[idx].onmessage = ({ data: { error, knnDistances, knnIndexes } }) => {\n                    if (error) {\n                        workers[idx].terminate();\n                        rejectWorker(error);\n                    }\n                    else {\n                        workers[idx].terminate();\n                        resolveWorker({ knnDistances, knnIndexes });\n                    }\n                };\n            });\n        }\n        const results = await Promise.all(promises);\n        const knnSizes = new Int32Array(values[0].length);\n        for (const res of results) {\n            for (let i = 0; i < values[0].length; ++i)\n                knnSizes[i] += res.knnIndexes[i]?.length ?? 0;\n        }\n        const knnRes = {\n            knnDistances: new Array(values[0].length).fill(null).map((_, i) => new Array(knnSizes[i])),\n            knnIndexes: new Array(values[0].length).fill(null).map((_, i) => new Array(knnSizes[i]))\n        };\n        for (const res of results) {\n            for (let i = 0; i < values[0].length; ++i) {\n                for (let j = 0; j < res.knnDistances[i]?.length ?? 0; ++j) {\n                    knnRes.knnDistances[i][knnSizes[i] - 1] = res.knnDistances[i][j];\n                    knnRes.knnIndexes[i][knnSizes[i] - 1] = res.knnIndexes[i][j];\n                    knnSizes[i] -= 1;\n                }\n            }\n        }\n        return knnRes;\n    }\n    async multiColumnKNN(values, fnNames, nNeighbours = 15, opts, weights, aggregationMethod = DistanceAggregationMethods.EUCLIDEAN) {\n        if (values.length !== fnNames.length || values.length !== opts.length || values.length !== weights.length)\n            throw new Error('values, distance functions, options and weights arrays should have the same length');\n        if (values.some((v) => v.length !== values[0].length))\n            throw new Error('all values arrays should have the same length');\n        const matSize = values[0].length * (values[0].length - 1) / 2;\n        const chunkSize = Math.floor(matSize / this._workerCount);\n        const promises = new Array(this._workerCount);\n        const workers = new Array(this._workerCount)\n            .fill(null).map(() => new Worker(new URL('./knn-worker', import.meta.url)));\n        for (let idx = 0; idx < this._workerCount; idx++) {\n            promises[idx] = new Promise((resolveWorker, rejectWorker) => {\n                const startIdx = idx * chunkSize;\n                const endIdx = idx === this._workerCount - 1 ? matSize : (idx + 1) * chunkSize;\n                if (endIdx <= startIdx)\n                    resolveWorker({ knnDistances: new Array(0), knnIndexes: new Array(0) });\n                workers[idx].postMessage({ values, startIdx, endIdx, fnNames, opts, nNeighbours, weights, aggregationMethod });\n                workers[idx].onmessage = ({ data: { error, knnDistances, knnIndexes } }) => {\n                    if (error) {\n                        workers[idx].terminate();\n                        rejectWorker(error);\n                    }\n                    else {\n                        workers[idx].terminate();\n                        resolveWorker({ knnDistances, knnIndexes });\n                    }\n                };\n            });\n        }\n        const results = await Promise.all(promises);\n        const knnRes = {\n            knnDistances: new Array(values[0].length).fill(null).map(() => new Array(nNeighbours).fill(99999)),\n            knnIndexes: new Array(values[0].length).fill(null).map(() => new Array(nNeighbours).fill(-1))\n        };\n        for (const res of results) {\n            for (let i = 0; i < values[0].length; ++i) {\n                for (let j = 0; j < res.knnDistances[i]?.length ?? 0; ++j)\n                    insertSmaller(knnRes.knnDistances[i], knnRes.knnIndexes[i], res.knnDistances[i][j], res.knnIndexes[i][j]);\n            }\n        }\n        return knnRes;\n    }\n    async getSampleDistances(values, fnNames, opts = [], weights, aggregationMethod = DistanceAggregationMethods.EUCLIDEAN) {\n        const thresholdWorkers = new Array(this._workerCount).fill(null)\n            .map(() => new Worker(new URL('./sparse-matrix-threshold-worker', import.meta.url)));\n        try {\n            const matSize = values[0].length * (values[0].length - 1) / 2;\n            const chunkSize = Math.floor(matSize / this._workerCount);\n            const maxSampleSize = 1000000;\n            const sampleSise = Math.max(Math.min(matSize / 1000, maxSampleSize), Math.min(matSize, maxSampleSize));\n            const testSetSizePerWorker = Math.floor(sampleSise / this._workerCount);\n            const tPromises = new Array(this._workerCount);\n            for (let idx = 0; idx < this._workerCount; idx++) {\n                tPromises[idx] = new Promise((resolveWorker, rejectWorker) => {\n                    const startIdx = idx * chunkSize;\n                    const endIdx = idx === this._workerCount - 1 ? matSize : (idx + 1) * chunkSize;\n                    thresholdWorkers[idx].postMessage({\n                        values: values, startIdx, endIdx, sampleLength: testSetSizePerWorker,\n                        fnNames, opts, weights, aggregationMethod\n                    });\n                    thresholdWorkers[idx].onmessage = ({ data: { error, distance } }) => {\n                        thresholdWorkers[idx].terminate();\n                        if (error)\n                            rejectWorker(error);\n                        else\n                            resolveWorker({ distance });\n                    };\n                });\n            }\n            const results = await Promise.all(tPromises);\n            const fullSize = results.reduce((acc, val) => acc + val.distance.length, 0);\n            const distance = new Float32Array(fullSize);\n            let offset = 0;\n            for (const res of results) {\n                distance.set(res.distance, offset);\n                offset += res.distance.length;\n            }\n            distance.sort();\n            return distance;\n        }\n        catch (e) {\n            thresholdWorkers?.forEach((w) => w?.terminate());\n            console.error(e);\n            return new Float32Array(1).fill(0.5);\n        }\n    }\n    async getMinimalThreshold(values, fnNames, opts = [], weights, aggregationMethod = DistanceAggregationMethods.EUCLIDEAN) {\n        //We need to calculate the minimal threshold first,\n        //in order to get matrix such that it does not exceed the maximum size of 1GB\n        //we have 3 return arrays, each 4 bites per element, so if the maximum size of the matrix is 1GB,\n        const maxSparseMatrixSize = 70000000;\n        try {\n            const matSize = values.length * (values.length - 1) / 2;\n            const distance = await this.getSampleDistances(values, fnNames, opts, weights, aggregationMethod);\n            const fractionIndex = Math.floor(maxSparseMatrixSize / matSize * distance.length);\n            const threshold = 1 - distance[fractionIndex];\n            // threshold = Math.max(threshold, 0.3);\n            return threshold;\n        }\n        catch (e) {\n            console.error(e);\n            return 0.5;\n        }\n    }\n    static calcSync(values, fnName, distanceFn, threshold) {\n        const i = [];\n        const j = [];\n        const distances = [];\n        let cnt = 0;\n        let mi = 0;\n        let mj = 0;\n        const fullSize = values.length * (values.length - 1) / 2;\n        while (cnt < fullSize) {\n            //const value = seq1List[mi] && seq1List[mj] ? hamming(seq1List[mi], seq1List[mj]) : 0;\n            const value = !isNil(values[mi]) && !isNil(values[mj]) ?\n                distanceFn(values[mi], values[mj]) : 1;\n            const similarity = 1 - value;\n            if (similarity >= threshold) {\n                i.push(mi);\n                j.push(mj);\n                distances.push(value);\n            }\n            cnt++;\n            mj++;\n            if (mj === values.length) {\n                mi++;\n                mj = mi + 1;\n            }\n        }\n        const iArray = new Int32Array(i);\n        const jArray = new Int32Array(j);\n        const distanceArray = new Float32Array(distances);\n        return { i: iArray, j: jArray, distance: distanceArray };\n    }\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sparse-matrix-service.js","sourceRoot":"","sources":["sparse-matrix-service.ts"],"names":[],"mappings":"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","sourcesContent":["import {KnownMetrics} from '../typed-metrics';\nimport {DistanceAggregationMethod, DistanceAggregationMethods} from './types';\nimport {insertSmaller, isNil} from './utils';\n\nexport type SparseMatrixResult = {\n  i: Int32Array | Uint32Array,\n  j: Int32Array | Uint32Array,\n  distance: Float32Array,\n  idx?: number\n};\n\nexport type KnnResult = {\n  knnDistances: number[][],\n  knnIndexes: number[][]\n}\nexport class SparseMatrixService {\n    private _workerCount: number;\n    constructor() {\n      this._workerCount = Math.max(navigator.hardwareConcurrency - 2, 1);\n    }\n\n    public async calcMultiColumn(values: Array<any[]>, fnNames: KnownMetrics[],\n      threshold: number, opts: {[_: string]: any}[] = [{}], weights: number[] = [1],\n      aggregationMethod: DistanceAggregationMethod = DistanceAggregationMethods.EUCLIDEAN\n    ) {\n      const matSize = values[0].length * (values[0].length - 1) / 2;\n      const chunkSize = Math.floor(matSize / this._workerCount);\n\n      const minThreshold = values[0].length > 20_000 ?\n        await this.getMinimalThreshold(values, fnNames, opts, weights, aggregationMethod) : 0;\n      if (threshold < minThreshold) {\n        console.log(`using threshold ${minThreshold}`);\n        threshold = minThreshold;\n      }\n      opts.forEach((_, i) => opts[i]['threshold'] = threshold);\n      const promises =\n        new Array<Promise<SparseMatrixResult>>(this._workerCount);\n\n      const workers = new Array(this._workerCount)\n        .fill(null).map(() => new Worker(new URL('./sparse-matrix-worker', import.meta.url)));\n      for (let idx = 0; idx < this._workerCount; idx++) {\n        promises[idx] = new Promise((resolveWorker, rejectWorker) => {\n          const startIdx = idx * chunkSize;\n          const endIdx = idx === this._workerCount - 1 ? matSize : (idx + 1) * chunkSize;\n          if (endIdx <= startIdx)\n            resolveWorker({i: new Int32Array(0), j: new Int32Array(0), distance: new Float32Array(0), idx});\n          workers[idx].postMessage({values, startIdx, endIdx, threshold, fnNames, opts, weights, aggregationMethod});\n          workers[idx].onmessage = ({data: {error, i, j, distance}}): void => {\n            if (error) {\n              workers[idx].terminate();\n              rejectWorker(error);\n            } else {\n              workers[idx].terminate();\n              resolveWorker({i, j, distance, idx});\n            }\n          };\n        });\n      }\n\n      const results = await Promise.all(promises);\n      const fullSize = results.reduce((acc, val) => acc + val.i.length, 0);\n      const i = new Int32Array(fullSize);\n      const j = new Int32Array(fullSize);\n      const distance = new Float32Array(fullSize);\n      let offset = 0;\n      // setting the results\n      for (const res of results) {\n        i.set(res.i, offset);\n        j.set(res.j, offset);\n        distance.set(res.distance, offset);\n        offset += res.i.length;\n      }\n      return {i, j, distance};\n    }\n\n    public async calc<T>(values: Array<T>, fnName: KnownMetrics, threshold: number, opts: {[_: string]: any} = {}) {\n      //size of full matrix\n      return await this.calcMultiColumn([values], [fnName], threshold, [opts], [1]);\n    }\n\n    public async getKNN(\n      values: Array<any>, fnName: KnownMetrics, nNeighbours: number = 15, opts: {[_: string]: any} = {}\n    ) {\n      return await this.multiColumnKNN([values], [fnName], nNeighbours, [opts], [1]);\n    }\n\n    public async getThresholdKNN(\n      values: Array<any>, fnName: KnownMetrics, threshold: number = 0.8, opts: {[_: string]: any} = {}\n    ) {\n      return await this.multiColumnThresholdKnn([values], [fnName], threshold, [opts], [1]);\n    }\n\n    public async multiColumnThresholdKnn(values: Array<Array<any>>, fnNames: KnownMetrics[], threshold: number = 0.8,\n      opts: {[_: string]: any}[], weights: number[],\n      aggregationMethod: DistanceAggregationMethod = DistanceAggregationMethods.EUCLIDEAN\n    ) {\n      if (values.length !== fnNames.length || values.length !== opts.length || values.length !== weights.length)\n        throw new Error('values, distance functions, options and weights arrays should have the same length');\n\n      if (values.some((v) => v.length !== values[0].length))\n        throw new Error('all values arrays should have the same length');\n\n      const matSize = values[0].length * (values[0].length - 1) / 2;\n      const chunkSize = Math.floor(matSize / this._workerCount);\n      const promises =\n        new Array<Promise<KnnResult>>(this._workerCount);\n      const workers = new Array(this._workerCount)\n        .fill(null).map(() => new Worker(new URL('./knn-threshold-worker', import.meta.url)));\n      for (let idx = 0; idx < this._workerCount; idx++) {\n        promises[idx] = new Promise((resolveWorker, rejectWorker) => {\n          const startIdx = idx * chunkSize;\n          const endIdx = idx === this._workerCount - 1 ? matSize : (idx + 1) * chunkSize;\n          if (endIdx <= startIdx)\n            resolveWorker({knnDistances: new Array(0), knnIndexes: new Array(0)});\n          workers[idx].postMessage({values, startIdx, endIdx, fnNames, opts, threshold, weights, aggregationMethod});\n          workers[idx].onmessage = ({data: {error, knnDistances, knnIndexes}}): void => {\n            if (error) {\n              workers[idx].terminate();\n              rejectWorker(error);\n            } else {\n              workers[idx].terminate();\n              resolveWorker({knnDistances, knnIndexes});\n            }\n          };\n        });\n      }\n\n      const results = await Promise.all(promises);\n      const knnSizes = new Int32Array(values[0].length);\n      for (const res of results) {\n        for (let i = 0; i < values[0].length; ++i)\n          knnSizes[i] += res.knnIndexes[i]?.length ?? 0;\n      }\n      const knnRes: KnnResult = {\n        knnDistances: new Array(values[0].length).fill(null).map((_, i) => new Array<number>(knnSizes[i])),\n        knnIndexes: new Array(values[0].length).fill(null).map((_, i) => new Array<number>(knnSizes[i]))};\n      for (const res of results) {\n        for (let i = 0; i < values[0].length; ++i) {\n          for (let j = 0; j < res.knnDistances[i]?.length ?? 0; ++j) {\n            knnRes.knnDistances[i][knnSizes[i] - 1] = res.knnDistances[i][j];\n            knnRes.knnIndexes[i][knnSizes[i] - 1] = res.knnIndexes[i][j];\n            knnSizes[i] -= 1;\n          }\n        }\n      }\n      return knnRes;\n    }\n\n    public async multiColumnKNN(values: Array<Array<any>>, fnNames: KnownMetrics[], nNeighbours: number = 15,\n      opts: {[_: string]: any}[], weights: number[],\n      aggregationMethod: DistanceAggregationMethod = DistanceAggregationMethods.EUCLIDEAN\n    ) {\n      if (values.length !== fnNames.length || values.length !== opts.length || values.length !== weights.length)\n        throw new Error('values, distance functions, options and weights arrays should have the same length');\n\n      if (values.some((v) => v.length !== values[0].length))\n        throw new Error('all values arrays should have the same length');\n\n      const matSize = values[0].length * (values[0].length - 1) / 2;\n      const chunkSize = Math.floor(matSize / this._workerCount);\n      const promises =\n        new Array<Promise<KnnResult>>(this._workerCount);\n      const workers = new Array(this._workerCount)\n        .fill(null).map(() => new Worker(new URL('./knn-worker', import.meta.url)));\n      for (let idx = 0; idx < this._workerCount; idx++) {\n        promises[idx] = new Promise((resolveWorker, rejectWorker) => {\n          const startIdx = idx * chunkSize;\n          const endIdx = idx === this._workerCount - 1 ? matSize : (idx + 1) * chunkSize;\n          if (endIdx <= startIdx)\n            resolveWorker({knnDistances: new Array(0), knnIndexes: new Array(0)});\n          workers[idx].postMessage({values, startIdx, endIdx, fnNames, opts, nNeighbours, weights, aggregationMethod});\n          workers[idx].onmessage = ({data: {error, knnDistances, knnIndexes}}): void => {\n            if (error) {\n              workers[idx].terminate();\n              rejectWorker(error);\n            } else {\n              workers[idx].terminate();\n              resolveWorker({knnDistances, knnIndexes});\n            }\n          };\n        });\n      }\n\n      const results = await Promise.all(promises);\n      const knnRes: KnnResult = {\n        knnDistances: new Array(values[0].length).fill(null).map(() => new Array<number>(nNeighbours).fill(99999)),\n        knnIndexes: new Array(values[0].length).fill(null).map(() => new Array<number>(nNeighbours).fill(-1))};\n      for (const res of results) {\n        for (let i = 0; i < values[0].length; ++i) {\n          for (let j = 0; j < res.knnDistances[i]?.length ?? 0; ++j)\n            insertSmaller(knnRes.knnDistances[i], knnRes.knnIndexes[i], res.knnDistances[i][j], res.knnIndexes[i][j]);\n        }\n      }\n      return knnRes;\n    }\n\n    public async getSampleDistances(values: Array<any[]>,\n      fnNames: KnownMetrics[], opts: {[_: string]: any}[] = [], weights: number[],\n      aggregationMethod: DistanceAggregationMethod = DistanceAggregationMethods.EUCLIDEAN): Promise<Float32Array> {\n      const thresholdWorkers = new Array(this._workerCount).fill(null)\n        .map(() => new Worker(new URL('./sparse-matrix-threshold-worker', import.meta.url)));\n\n      try {\n        const matSize = values[0].length * (values[0].length - 1) / 2;\n        const chunkSize = Math.floor(matSize / this._workerCount);\n        const maxSampleSize = 1_000_000;\n        const sampleSise = Math.max(Math.min(matSize / 1000, maxSampleSize), Math.min(matSize, maxSampleSize));\n        const testSetSizePerWorker = Math.floor(sampleSise / this._workerCount);\n        const tPromises = new Array<Promise<{distance: Float32Array}>>(this._workerCount);\n\n        for (let idx = 0; idx < this._workerCount; idx++) {\n          tPromises[idx] = new Promise((resolveWorker, rejectWorker) => {\n            const startIdx = idx * chunkSize;\n            const endIdx = idx === this._workerCount - 1 ? matSize : (idx + 1) * chunkSize;\n            thresholdWorkers[idx].postMessage({\n              values: values, startIdx, endIdx, sampleLength: testSetSizePerWorker,\n              fnNames, opts, weights, aggregationMethod\n            });\n            thresholdWorkers[idx].onmessage = ({data: {error, distance}}): void => {\n              thresholdWorkers[idx].terminate();\n              if (error) rejectWorker(error); else\n                resolveWorker({distance});\n            };\n          });\n        }\n\n        const results = await Promise.all(tPromises);\n        const fullSize = results.reduce((acc, val) => acc + val.distance.length, 0);\n        const distance = new Float32Array(fullSize);\n        let offset = 0;\n        for (const res of results) {\n          distance.set(res.distance, offset);\n          offset += res.distance.length;\n        }\n        distance.sort();\n\n        return distance;\n      } catch (e) {\n        thresholdWorkers?.forEach((w) => w?.terminate());\n        console.error(e);\n        return new Float32Array(1).fill(0.5);\n      }\n    }\n\n    private async getMinimalThreshold(values: Array<any[]>,\n      fnNames: KnownMetrics[], opts: {[_: string]: any}[] = [], weights: number[],\n      aggregationMethod: DistanceAggregationMethod = DistanceAggregationMethods.EUCLIDEAN) {\n      //We need to calculate the minimal threshold first,\n      //in order to get matrix such that it does not exceed the maximum size of 1GB\n      //we have 3 return arrays, each 4 bites per element, so if the maximum size of the matrix is 1GB,\n      const maxSparseMatrixSize = 70_000_000;\n      try {\n        const matSize = values.length * (values.length - 1) / 2;\n        const distance = await this.getSampleDistances(values, fnNames, opts, weights, aggregationMethod);\n        const fractionIndex = Math.floor(maxSparseMatrixSize / matSize * distance.length);\n        const threshold = 1 - distance[fractionIndex];\n        // threshold = Math.max(threshold, 0.3);\n        return threshold;\n      } catch (e) {\n        console.error(e);\n        return 0.5;\n      }\n    }\n\n    public static calcSync<T>(\n      values: Array<T> | ArrayLike<T>, fnName: KnownMetrics, distanceFn: Function, threshold: number\n    ) {\n      const i: number[] = [];\n      const j: number[] = [];\n      const distances: number[] = [];\n      let cnt = 0;\n      let mi = 0;\n      let mj = 0;\n      const fullSize = values.length * (values.length - 1) / 2;\n      while (cnt < fullSize) {\n        //const value = seq1List[mi] && seq1List[mj] ? hamming(seq1List[mi], seq1List[mj]) : 0;\n        const value = !isNil(values[mi]) && !isNil(values[mj]) ?\n          distanceFn(values[mi], values[mj]) : 1;\n        const similarity = 1 - value;\n        if (similarity >= threshold) {\n          i.push(mi);\n          j.push(mj);\n          distances.push(value);\n        }\n        cnt++;\n        mj++;\n        if (mj === values.length) {\n          mi++;\n          mj = mi + 1;\n        }\n      }\n\n      const iArray = new Int32Array(i);\n      const jArray = new Int32Array(j);\n      const distanceArray = new Float32Array(distances);\n\n      return {i: iArray, j: jArray, distance: distanceArray};\n    }\n}\n"]}","export async function bioLayout(cluster, sparseObject, threshold) {\n    const n = cluster.length;\n    const is = [];\n    const js = [];\n    for (let i = 0; i < n; i++) {\n        for (let j = i + 1; j < n; j++) {\n            const a = cluster[i];\n            const b = cluster[j];\n            if (sparseObject[a]?.[b] >= threshold) {\n                is.push(i);\n                js.push(j);\n            }\n        }\n    }\n    return getLayoutEmbeddings(is, js, cluster);\n}\nfunction getLayoutEmbeddings(is, js, cluster) {\n    const embedX = new Float32Array(cluster.length).fill(0).map(() => Math.random() * 10);\n    const embedY = new Float32Array(cluster.length).fill(0).map(() => Math.random() * 10);\n    const iterations = 100;\n    const xVelocities = new Float32Array(cluster.length).fill(0);\n    const yVelocities = new Float32Array(cluster.length).fill(0);\n    for (let i = 0; i < iterations; i++) {\n        const temperature = (1 - i / iterations);\n        xVelocities.fill(0);\n        yVelocities.fill(0);\n        for (let idx = 0; idx < is.length; idx++) {\n            const a = is[idx];\n            const b = js[idx];\n            const dx = embedX[a] - embedX[b];\n            const dy = embedY[a] - embedY[b];\n            //   const distance = Math.sqrt(dx * dx + dy * dy);\n            //   const factor = (distance - 1) / distance;\n            //   const offsetX = dx * factor;\n            //   const offsetY = dy * factor;\n            if (Math.abs(dx) >= 1) {\n                xVelocities[a] -= temperature * dx;\n                xVelocities[b] += temperature * dx;\n            }\n            if (Math.abs(dy) >= 1) {\n                yVelocities[a] -= temperature * dy;\n                yVelocities[b] += temperature * dy;\n            }\n            //   embedY[a] += offsetY * temperature;\n            //   embedY[b] -= offsetY * temperature;\n        }\n        for (let idx = 0; idx < cluster.length; idx++) {\n            const vecSize = Math.sqrt(xVelocities[idx] * xVelocities[idx] + yVelocities[idx] * yVelocities[idx]);\n            if (vecSize > 0) {\n                embedX[idx] += xVelocities[idx] / vecSize * temperature;\n                embedY[idx] += yVelocities[idx] / vecSize * temperature;\n            }\n        }\n    }\n    // normalize\n    let minX = embedX[0];\n    let minY = embedY[0];\n    let maxX = embedX[0];\n    let maxY = embedY[0];\n    for (let idx = 1; idx < cluster.length; idx++) {\n        minX = Math.min(minX, embedX[idx]);\n        minY = Math.min(minY, embedY[idx]);\n        maxX = Math.max(maxX, embedX[idx]);\n        maxY = Math.max(maxY, embedY[idx]);\n    }\n    let rangeX = maxX - minX;\n    let rangeY = maxY - minY;\n    if (rangeX === 0)\n        rangeX = maxX;\n    for (let idx = 0; idx < cluster.length; idx++)\n        embedX[idx] = (embedX[idx] - minX) / rangeX / 2 + 0.5;\n    if (rangeY === 0)\n        rangeY = maxY;\n    for (let idx = 0; idx < cluster.length; idx++)\n        embedY[idx] = (embedY[idx] - minY) / rangeY / 2 + 0.5;\n    return { embedX, embedY };\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bio-layout.js","sourceRoot":"","sources":["bio-layout.ts"],"names":[],"mappings":"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","sourcesContent":["import {SparseMatrixObject} from './types';\n\nexport async function bioLayout(cluster: number[], sparseObject: SparseMatrixObject, threshold: number) {\n  const n = cluster.length;\n  const is: number[] = [];\n  const js: number[] = [];\n  for (let i = 0; i < n; i++) {\n    for (let j = i + 1; j < n; j++) {\n      const a = cluster[i];\n      const b = cluster[j];\n      if (sparseObject[a]?.[b] >= threshold) {\n        is.push(i);\n        js.push(j);\n      }\n    }\n  }\n  return getLayoutEmbeddings(is, js, cluster);\n}\n\nfunction getLayoutEmbeddings(is: number[], js: number[], cluster: number[]) {\n  const embedX = new Float32Array(cluster.length).fill(0).map(() => Math.random() * 10);\n  const embedY = new Float32Array(cluster.length).fill(0).map(() => Math.random() * 10);\n  const iterations = 100;\n  const xVelocities = new Float32Array(cluster.length).fill(0);\n  const yVelocities = new Float32Array(cluster.length).fill(0);\n  for (let i = 0; i < iterations; i++) {\n    const temperature = (1 - i / iterations);\n    xVelocities.fill(0);\n    yVelocities.fill(0);\n    for (let idx = 0; idx < is.length; idx++) {\n      const a = is[idx];\n      const b = js[idx];\n      const dx = embedX[a] - embedX[b];\n      const dy = embedY[a] - embedY[b];\n      //   const distance = Math.sqrt(dx * dx + dy * dy);\n      //   const factor = (distance - 1) / distance;\n      //   const offsetX = dx * factor;\n      //   const offsetY = dy * factor;\n      if (Math.abs(dx) >= 1) {\n        xVelocities[a] -= temperature * dx;\n        xVelocities[b] += temperature * dx;\n      }\n      if (Math.abs(dy) >= 1) {\n        yVelocities[a] -= temperature * dy;\n        yVelocities[b] += temperature * dy;\n      }\n    //   embedY[a] += offsetY * temperature;\n    //   embedY[b] -= offsetY * temperature;\n    }\n    for (let idx = 0; idx < cluster.length; idx++) {\n      const vecSize = Math.sqrt(xVelocities[idx] * xVelocities[idx] + yVelocities[idx] * yVelocities[idx]);\n      if (vecSize > 0) {\n        embedX[idx] += xVelocities[idx] / vecSize * temperature;\n        embedY[idx] += yVelocities[idx] / vecSize * temperature;\n      }\n    }\n  }\n\n  // normalize\n  let minX = embedX[0];\n  let minY = embedY[0];\n  let maxX = embedX[0];\n  let maxY = embedY[0];\n  for (let idx = 1; idx < cluster.length; idx++) {\n    minX = Math.min(minX, embedX[idx]);\n    minY = Math.min(minY, embedY[idx]);\n    maxX = Math.max(maxX, embedX[idx]);\n    maxY = Math.max(maxY, embedY[idx]);\n  }\n  let rangeX = maxX - minX;\n  let rangeY = maxY - minY;\n  if (rangeX === 0)\n    rangeX = maxX;\n  for (let idx = 0; idx < cluster.length; idx++)\n    embedX[idx] = (embedX[idx] - minX) / rangeX / 2 + 0.5;\n  if (rangeY === 0)\n    rangeY = maxY;\n  for (let idx = 0; idx < cluster.length; idx++)\n    embedY[idx] = (embedY[idx] - minY) / rangeY / 2 + 0.5;\n\n  return {embedX, embedY};\n}\n"]}","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\n/**\n * colwise normalize the knn similarities\n * BE WARE, modifies the input array in place\n * @param device\n * @param knnSimilarities\n * @param offsets\n * @param nRows\n */\nexport function sparseKNNNoralizeColwise(device, knnSimilarities, offsets, nRows) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const neededThreads = nRows;\n        const workGroupThreadsPerDim = 10;\n        const totalWorkgroupThreads = workGroupThreadsPerDim * workGroupThreadsPerDim;\n        const neededWorkGroups = Math.ceil(neededThreads / totalWorkgroupThreads);\n        const workGroupDim = Math.ceil(Math.sqrt(neededWorkGroups));\n        const threadsPerDim = workGroupThreadsPerDim * workGroupDim;\n        const module = device.createShaderModule({\n            label: 'colwise-normalize',\n            code: `\n        @group(0) @binding(0) var<storage, read_write> knnSimilarities : array<f32>;\n        @group(0) @binding(1) var<storage, read> offsets : array<u32>;\n        @compute @workgroup_size(${workGroupThreadsPerDim}, ${workGroupThreadsPerDim}) fn normalize(\n            @builtin(global_invocation_id) id: vec3<u32>,\n          ) {\n            let row = id.x;\n            let col = id.y;\n            let index = row * ${threadsPerDim} + col;\n            if (index >= ${nRows}) {\n                return;\n            }\n            let offsetBegin = offsets[index];\n            let offsetEnd = offsets[index + 1];\n            var sum = 0.0;\n            if (offsetEnd - offsetBegin == 0) {\n                return;\n            }\n            for (var i = offsetBegin; i < offsetEnd; i = i + 1) {\n                sum = sum + knnSimilarities[i];\n            }\n            if (sum > 0.0) {\n              for (var i = offsetBegin; i < offsetEnd; i = i + 1) {\n                  knnSimilarities[i] = knnSimilarities[i] / sum;\n              }\n            }\n          }\n        \n        `,\n        });\n        const pipeline = device.createComputePipeline({\n            label: 'hamming compute pipeline',\n            layout: 'auto',\n            compute: {\n                module,\n                entryPoint: 'normalize',\n            },\n        });\n        const simmilaritiesBuffer = device.createBuffer({\n            label: 'simmilarities info buffer',\n            size: knnSimilarities.byteLength,\n            usage: GPUBufferUsage.STORAGE |\n                GPUBufferUsage.COPY_SRC |\n                GPUBufferUsage.COPY_DST,\n            mappedAtCreation: true,\n        });\n        new Float32Array(simmilaritiesBuffer.getMappedRange()).set(knnSimilarities);\n        simmilaritiesBuffer.unmap();\n        const offsetsBuffer = device.createBuffer({\n            label: 'offsets info buffer',\n            size: offsets.byteLength,\n            usage: GPUBufferUsage.STORAGE |\n                GPUBufferUsage.COPY_SRC |\n                GPUBufferUsage.COPY_DST,\n            mappedAtCreation: true,\n        });\n        new Uint32Array(offsetsBuffer.getMappedRange()).set(offsets);\n        offsetsBuffer.unmap();\n        const bindGroup = device.createBindGroup({\n            layout: pipeline.getBindGroupLayout(0),\n            entries: [\n                {\n                    binding: 0,\n                    resource: {\n                        buffer: simmilaritiesBuffer,\n                    },\n                },\n                {\n                    binding: 1,\n                    resource: {\n                        buffer: offsetsBuffer,\n                    },\n                },\n            ],\n        });\n        const commandEncoder = device.createCommandEncoder();\n        const passEncoder = commandEncoder.beginComputePass();\n        passEncoder.setPipeline(pipeline);\n        passEncoder.setBindGroup(0, bindGroup);\n        passEncoder.dispatchWorkgroups(workGroupDim, workGroupDim);\n        passEncoder.end();\n        const outSimilarityBuffer = device.createBuffer({\n            label: 'out similarity buffer',\n            size: knnSimilarities.byteLength,\n            usage: GPUBufferUsage.MAP_READ | GPUBufferUsage.COPY_DST,\n        });\n        commandEncoder.copyBufferToBuffer(simmilaritiesBuffer, 0, outSimilarityBuffer, 0, knnSimilarities.byteLength);\n        device.queue.submit([commandEncoder.finish()]);\n        yield device.queue.onSubmittedWorkDone();\n        yield outSimilarityBuffer.mapAsync(GPUMapMode.READ);\n        const arrayBuffer = new Float32Array(outSimilarityBuffer.getMappedRange());\n        knnSimilarities.set(arrayBuffer);\n        outSimilarityBuffer.unmap();\n        simmilaritiesBuffer.destroy();\n        offsetsBuffer.destroy();\n        outSimilarityBuffer.destroy();\n    });\n}\n//# sourceMappingURL=colwise-normalize.js.map","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\n/**\n * inflation operator in place\n */\nexport function inflate(knnSimilarities, factor = 2) {\n    for (let i = 0; i < knnSimilarities.length; i++)\n        knnSimilarities[i] = Math.pow(knnSimilarities[i], factor);\n}\nfunction getRowIndexes(offsets) {\n    const res = new Uint32Array(offsets[offsets.length - 1]);\n    for (let i = 0; i < offsets.length - 1; i++) {\n        for (let j = offsets[i]; j < offsets[i + 1]; j++)\n            res[j] = i;\n    }\n    return res;\n}\n// this implementation is not looking at already zeroed out values, and only looks at live cells.\nexport function expandNoRevive(device, knnSimilarities, knnIndexes, offsets, nRows) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const neededThreads = 90000;\n        const workGroupThreadsPerDim = 10;\n        const totalWorkgroupThreads = workGroupThreadsPerDim * workGroupThreadsPerDim;\n        const neededWorkGroups = Math.ceil(neededThreads / totalWorkgroupThreads);\n        const workGroupDim = Math.ceil(Math.sqrt(neededWorkGroups));\n        const threadsPerDim = workGroupThreadsPerDim * workGroupDim;\n        const order = Math.floor(Math.max(Math.log10(nRows), 2)) + 1;\n        // minimum value after expansion.\n        const pruneValue = Math.pow(10, -order);\n        const outKNNSimilarities = new Float32Array(knnSimilarities.length);\n        const module = device.createShaderModule({\n            label: 'expand',\n            code: `\n        struct SparseKNN {\n            knnSimilarities: array<f32, ${knnSimilarities.length}>,\n            knnIndexes: array<u32, ${knnIndexes.length}>,\n            offsets: array<u32, ${offsets.length}>,\n            rowIndexes: array<u32, ${knnIndexes.length}>,\n        }\n\n        @group(0) @binding(0) var<storage, read_write> sparseKNN: SparseKNN;\n        @group(0) @binding(1) var<storage, read_write> resultSimBlock: array<f32, ${neededThreads}>;\n        @group(0) @binding(2) var<storage, read_write> startAt: u32;\n        @compute @workgroup_size(${workGroupThreadsPerDim}, ${workGroupThreadsPerDim}) fn expand(\n            @builtin(global_invocation_id) id: vec3<u32>,\n          ) {\n            let col = id.x;\n            let row = id.y;\n            let index = row * ${threadsPerDim} + col;\n            if (index >= ${neededThreads}) {\n                return;\n            }\n            let workingIndex = index + startAt;\n            if (workingIndex >= ${knnIndexes.length}) {\n                return;\n            }\n            \n            let rowIdx = sparseKNN.rowIndexes[workingIndex];\n            let colIdx = sparseKNN.knnIndexes[workingIndex];\n            let offsetBeginRow = sparseKNN.offsets[rowIdx];\n            let offsetEndRow = sparseKNN.offsets[rowIdx + 1];\n            let offsetBeginCol = sparseKNN.offsets[colIdx];\n            let offsetEndCol = sparseKNN.offsets[colIdx + 1];\n            var sum = 0.0;\n            for (var i = offsetBeginRow; i < offsetEndRow; i = i + 1) {\n                for(var j = offsetBeginCol; j < offsetEndCol; j = j + 1) {\n                    if (sparseKNN.knnIndexes[i] == sparseKNN.knnIndexes[j]) {\n                        sum = sum + sparseKNN.knnSimilarities[i] * sparseKNN.knnSimilarities[j];\n                        break;\n                    }\n                }\n            }\n            if (sum > ${pruneValue}) {\n                resultSimBlock[index] = sum;\n            } else {\n                resultSimBlock[index] = 0.0;\n            }\n        }\n    `\n        });\n        const pipeline = device.createComputePipeline({\n            label: 'expand compute pipeline',\n            layout: 'auto',\n            compute: {\n                module: module,\n                entryPoint: 'expand',\n            },\n        });\n        const sparseKNNBuffer32Size = knnSimilarities.length + knnIndexes.length + offsets.length + knnIndexes.length;\n        let sparseKNNBufferByteSize = sparseKNNBuffer32Size * 4;\n        const remainder = sparseKNNBufferByteSize & 15;\n        if (remainder !== 0)\n            sparseKNNBufferByteSize += 16 - remainder;\n        const sparseKNNBuffer = device.createBuffer({\n            label: 'sparse knn buffer',\n            size: sparseKNNBufferByteSize,\n            usage: GPUBufferUsage.STORAGE |\n                GPUBufferUsage.COPY_SRC |\n                GPUBufferUsage.COPY_DST,\n            mappedAtCreation: true,\n        });\n        const sparseKNNArrayBuffer = sparseKNNBuffer.getMappedRange();\n        // set similarities\n        new Float32Array(sparseKNNArrayBuffer, 0, knnSimilarities.length).set(knnSimilarities);\n        // set indexes\n        new Uint32Array(sparseKNNArrayBuffer, knnSimilarities.length * 4, knnIndexes.length).set(knnIndexes);\n        // set offsets\n        new Uint32Array(sparseKNNArrayBuffer, (knnSimilarities.length + knnIndexes.length) * 4, offsets.length).set(offsets);\n        // set row indexes\n        const rowIndexes = getRowIndexes(offsets);\n        new Uint32Array(sparseKNNArrayBuffer, (knnSimilarities.length + knnIndexes.length + offsets.length) * 4, rowIndexes.length).set(rowIndexes);\n        sparseKNNBuffer.unmap();\n        const startBuffer = device.createBuffer({\n            label: 'start end buffer',\n            size: 4,\n            usage: GPUBufferUsage.STORAGE |\n                GPUBufferUsage.COPY_SRC |\n                GPUBufferUsage.COPY_DST,\n            mappedAtCreation: true,\n        });\n        new Uint32Array(startBuffer.getMappedRange()).set([0]);\n        startBuffer.unmap();\n        const resultBlockBuffer = device.createBuffer({\n            label: 'result block buffer',\n            size: neededThreads * 4,\n            usage: GPUBufferUsage.STORAGE |\n                GPUBufferUsage.COPY_SRC |\n                GPUBufferUsage.COPY_DST,\n        });\n        const bindGroup = device.createBindGroup({\n            layout: pipeline.getBindGroupLayout(0),\n            entries: [\n                { binding: 0, resource: { buffer: sparseKNNBuffer } },\n                { binding: 1, resource: { buffer: resultBlockBuffer } },\n                { binding: 2, resource: { buffer: startBuffer } },\n            ],\n        });\n        const outBlockBuffer = device.createBuffer({\n            label: 'out block buffer',\n            size: resultBlockBuffer.size,\n            usage: GPUBufferUsage.MAP_READ | GPUBufferUsage.COPY_DST\n        });\n        for (let i = 0; i < Math.ceil(knnIndexes.length / neededThreads); i++) {\n            const start = i * neededThreads;\n            const curLength = Math.min(neededThreads, knnIndexes.length - start);\n            device.queue.writeBuffer(startBuffer, 0, new Uint32Array([start]));\n            const encoder = device.createCommandEncoder({\n                label: 'expand encoder',\n            });\n            const pass = encoder.beginComputePass({\n                label: 'expand compute pass',\n            });\n            pass.setPipeline(pipeline);\n            pass.setBindGroup(0, bindGroup);\n            pass.dispatchWorkgroups(workGroupDim, workGroupDim);\n            pass.end();\n            encoder.copyBufferToBuffer(resultBlockBuffer, 0, outBlockBuffer, 0, outBlockBuffer.size);\n            device.queue.submit([encoder.finish()]);\n            //console.time('onSubmittedWorkDone');\n            yield device.queue.onSubmittedWorkDone();\n            //console.timeEnd('onSubmittedWorkDone');\n            yield outBlockBuffer.mapAsync(GPUMapMode.READ);\n            const outBlock = new Float32Array(outBlockBuffer.getMappedRange(), 0, curLength);\n            //console.time('fillout');\n            outKNNSimilarities.set(outBlock, start);\n            //console.timeEnd('fillout');\n            outBlockBuffer.unmap();\n        }\n        // destroy\n        sparseKNNBuffer.destroy();\n        startBuffer.destroy();\n        resultBlockBuffer.destroy();\n        outBlockBuffer.destroy();\n        return { KNNIndexes: knnIndexes, KNNSimilarities: outKNNSimilarities, indexOffsets: offsets };\n    });\n}\n//# sourceMappingURL=inflate-expand.js.map","import { toOffsetForm } from '../umap/utils';\nexport function countOccurancesAddLoops(sparseMatrix, nRows) {\n    const res = new Uint32Array(nRows);\n    for (let i = 0; i < sparseMatrix.i.length; i++) {\n        res[sparseMatrix.i[i]]++;\n        res[sparseMatrix.j[i]]++;\n    }\n    // add loops\n    for (let i = 0; i < nRows; i++)\n        res[i]++;\n    return res;\n}\n/**\n * BE WARE, this function is ment for half sparse matrices only!!!\n * @param sparseMatrix\n * @param nRows\n * @returns\n */\nexport function toSparseKNNSimilarityForm(sparseMatrix, nRows) {\n    const occurenceCount = countOccurancesAddLoops(sparseMatrix, nRows);\n    const offsetForm = toOffsetForm(occurenceCount);\n    const insertCounter = new Uint32Array(nRows).fill(1); // for counting how many things we inserted into the knn arrays.\n    // we initialize these counters to one because first we insert the self loops\n    // dont be confused by the name knn, k here is variable, its just the thing that in this form, we know that at index i\n    // we have some k(variable) ammount of sparse neigbours\n    // we know for a fact that sparse matrix will not contain any duplicates, we add nRows to account for self loops\n    const KNNIndexes = new Uint32Array(sparseMatrix.i.length * 2 + nRows);\n    const KNNSimilarities = new Float32Array(sparseMatrix.i.length * 2 + nRows);\n    // insert self loops at the start of each offset\n    for (let i = 0; i < nRows; i++) {\n        KNNIndexes[offsetForm[i]] = i;\n        KNNSimilarities[offsetForm[i]] = 1;\n    }\n    for (let i = 0; i < sparseMatrix.i.length; i++) {\n        const row = sparseMatrix.i[i];\n        const col = sparseMatrix.j[i];\n        const similarity = 1 - sparseMatrix.distance[i];\n        KNNIndexes[offsetForm[row] + insertCounter[row]] = col;\n        KNNSimilarities[offsetForm[row] + insertCounter[row]] = similarity;\n        insertCounter[row]++;\n        KNNIndexes[offsetForm[col] + insertCounter[col]] = row;\n        KNNSimilarities[offsetForm[col] + insertCounter[col]] = similarity;\n        insertCounter[col]++;\n    }\n    return { KNNIndexes, KNNSimilarities, indexOffsets: offsetForm };\n}\nexport function knnToSparseForm(knnIndexes, knnSims) {\n    const lengths = knnIndexes.map((x) => x.length);\n    const totalLength = lengths.reduce((a, b) => a + b, 0);\n    const offsets = toOffsetForm(lengths);\n    const outIndexes = new Uint32Array(totalLength);\n    const outSims = new Float32Array(totalLength);\n    for (let i = 0; i < knnIndexes.length; i++) {\n        outIndexes.set(knnIndexes[i], offsets[i]);\n        outSims.set(knnSims[i], offsets[i]);\n    }\n    return { KNNIndexes: outIndexes, KNNSimilarities: outSims, indexOffsets: offsets };\n}\n//# sourceMappingURL=utils.js.map","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\nimport { levenbergMarquardt } from 'ml-levenberg-marquardt';\nexport function linear(x1, x2, n) {\n    const step = (x2 - x1) / (n - 1);\n    return Array.from({ length: n }, (_, i) => x1 + step * i);\n}\n/**\n * Fit a, b params for the differentiable curve used in lower\n * dimensional fuzzy simplicial complex construction. We want the\n * smooth curve (from a pre-defined family with simple gradient) that\n * best matches an offset exponential decay.\n */\nexport function findABParams(spread, minDist) {\n    const curve = ([a, b]) => (x) => {\n        return 1.0 / (1.0 + a * Math.pow(x, (2 * b)));\n    };\n    const xv = linear(0, spread * 3, 300)\n        .map((val) => (val < minDist ? 1.0 : val));\n    const yv = new Array(xv.length).fill(0).map((val, index) => {\n        const gte = xv[index] >= minDist;\n        return gte ? Math.exp(-(xv[index] - minDist) / spread) : val;\n    });\n    const initialValues = [0.5, 0.5];\n    const data = { x: xv, y: yv };\n    // Default options for the algorithm (from github example)\n    const options = {\n        damping: 1.5,\n        initialValues,\n        gradientDifference: 10e-2,\n        maxIterations: 100,\n        errorTolerance: 10e-3,\n    };\n    // eslint-disable-next-line new-cap\n    const { parameterValues } = levenbergMarquardt(data, curve, options);\n    const [a, b] = parameterValues;\n    return { a, b };\n}\n/**\n * heuristic for number of epochs of umap, differs from cpu version, because of the way gpu passes data\n * @param entryLen number of entries\n * @returns\n */\nexport function getNEpochs(entryLen) {\n    //\n    const length = entryLen;\n    if (length <= 2500)\n        return 1000;\n    else if (length <= 5000)\n        return 800;\n    else if (length <= 7500)\n        return 650;\n    else\n        return 400;\n}\n/**  helper function that transforms array of sizes to offset form */\nexport function toOffsetForm(ar) {\n    const res = new Uint32Array(ar.length + 1);\n    let offset = 0;\n    for (let i = 0; i < ar.length; i++) {\n        res[i] = offset;\n        offset += ar[i];\n    }\n    res[ar.length] = offset;\n    return res;\n}\n/** multiplies flat representation of sparse knn info by scalar.\n * also no point of doing this on gpu, can be done on cpu pretty quickly.\n * if there will be need for it, it can be done on gpu as well very very easily.\n * @param values values array\n * @param scalar\n */\nexport function multiplyScalar(values, scalar) {\n    for (let i = 0; i < values.length; i++)\n        values[i] = values[i] * scalar;\n}\n/**\n * generate transpose of knn from the transposition info taken from sparse knn info\n * using this information avoids billions of array.push calls, is much cleaner, allows typed array use and is faster.\n * no point of doing this on gpu, can be done on cpu as we already have all the info needed.\n * @param knnIndexes\n * @param knnDistances\n * @param knnSparseInfo\n * @param knnSize\n * @returns\n */\nexport function transposeKNN(knnIndexes, knnDistances, knnSparseInfo, knnSize = 15) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const numOfEntries = knnIndexes.length;\n        const fullKnnEntrySize = knnSize * numOfEntries;\n        const transposeSizes = knnSparseInfo.resultTransposeSizesArray;\n        const transposeOffsets = toOffsetForm(transposeSizes);\n        const offsetsCopy = new Uint32Array(transposeOffsets.length);\n        offsetsCopy.set(transposeOffsets); // will be used for dynamic indexing\n        const transposeKNNIndexes = new Int32Array(fullKnnEntrySize).fill(0);\n        const transposeKNNDistances = new Float32Array(fullKnnEntrySize).fill(0);\n        for (let i = 0; i < numOfEntries; i++) {\n            for (let j = 0; j < knnSize; j++) {\n                const otherIndex = knnIndexes[i][j];\n                const otherIndexOffset = offsetsCopy[otherIndex];\n                transposeKNNIndexes[otherIndexOffset] = i;\n                transposeKNNDistances[otherIndexOffset] = knnDistances[i][j];\n                offsetsCopy[otherIndex] += 1;\n            }\n        }\n        return { transposeKNNIndexes, transposeKNNDistances, transposeOffsets };\n    });\n}\n//# sourceMappingURL=utils.js.map","// eslint-disable-next-line @typescript-eslint/unbound-method\nconst toString = Object.prototype.toString;\n/**\n * Checks if an object is an instance of an Array (array or typed array, except those that contain bigint values).\n *\n * @param value - Object to check.\n * @returns True if the object is an array or a typed array.\n */\nexport function isAnyArray(value) {\n    const tag = toString.call(value);\n    return tag.endsWith('Array]') && !tag.includes('Big');\n}\n//# sourceMappingURL=index.js.map","import { bioLayout } from './bio-layout';\nimport { markovClusterWebGPU } from '@datagrok-libraries/math/src/webGPU/MCL/MCL-webGPU';\nexport const defaultMCLOptions = {\n    expandFactor: 2,\n    maxIterations: 5,\n    inflateFactor: 2,\n    multFactor: 1,\n};\nexport class MCLSparseReducer {\n    constructor(opts = {}) {\n        this._options = { ...defaultMCLOptions, ...opts };\n    }\n    async transform(sparseMatrix, nRows) {\n        // testWorkerMultiply();\n        // return new Int32Array(nRows);\n        let sparseObject = this.toObjectForm(sparseMatrix);\n        if (this._options.maxIterations > 0) {\n            this.addLoops(sparseObject, nRows);\n            this.normalize(sparseObject);\n            for (let i = 0; i < this._options.maxIterations; i++) {\n                sparseObject = this.expand(sparseObject, nRows);\n                this.inflate(sparseObject);\n                this.normalize(sparseObject);\n            }\n        }\n        const { clusters, is, js } = this.assignClusters(sparseObject, nRows);\n        this.correctClusters(clusters);\n        const embeddings = await this.layout(clusters, sparseObject, nRows);\n        return { clusters, embedX: embeddings.embedX, embedY: embeddings.embedY, is, js };\n    }\n    async transformWebGPU(sparseMatrix, nRows) {\n        if (this._options.maxIterations === 0)\n            return this.transform(sparseMatrix, nRows);\n        const mclRes = await markovClusterWebGPU(sparseMatrix, nRows, this._options.maxIterations, this._options.inflateFactor);\n        const sparseObject = this.csrToSparseObject(mclRes, nRows);\n        const { clusters, is, js } = this.assignClusters(sparseObject, nRows);\n        this.correctClusters(clusters);\n        const embeddings = await this.layout(clusters, sparseObject, nRows);\n        return { clusters, embedX: embeddings.embedX, embedY: embeddings.embedY, is, js };\n    }\n    correctClusters(clusters) {\n        const clusterSizeMap = {};\n        for (const cluster of clusters) {\n            if (!clusterSizeMap[cluster])\n                clusterSizeMap[cluster] = 0;\n            clusterSizeMap[cluster]++;\n        }\n        const sortedIndexes = Object.keys(clusterSizeMap).map(Number).sort((a, b) => clusterSizeMap[b] - clusterSizeMap[a]);\n        const clusterMap = {};\n        sortedIndexes.forEach((clusterIdx, i) => clusterMap[clusterIdx] = i + 1);\n        for (let i = 0; i < clusters.length; i++)\n            clusters[i] = clusterMap[clusters[i]];\n        // let curCluster = 1;\n        // for (let i = 0; i < clusters.length; i++) {\n        //   if (!clusterMap[clusters[i]]) {\n        //     clusterMap[clusters[i]] = curCluster;\n        //     clusters[i] = curCluster;\n        //     curCluster++;\n        //   } else {\n        //     clusters[i] = clusterMap[clusters[i]];\n        //   }\n        // }\n    }\n    csrToSparseObject(mclRes, nRows) {\n        const order = Math.floor(Math.max(Math.log10(nRows), 2)) + 1;\n        const minOrder = 1 / Math.pow(10, order);\n        const sparseObject = {};\n        for (let i = 0; i < nRows; i++) {\n            sparseObject[i] = {};\n            for (let k = mclRes.indexOffsets[i]; k < mclRes.indexOffsets[i + 1]; k++) {\n                const j = mclRes.KNNIndexes[k];\n                if (j <= i || mclRes.KNNSimilarities[k] < minOrder)\n                    continue;\n                sparseObject[i][j] = mclRes.KNNSimilarities[k];\n            }\n        }\n        return sparseObject;\n    }\n    async layout(clusters, sparseMatrix, nRows) {\n        const embedX = new Float32Array(nRows).fill(0);\n        const embedY = new Float32Array(nRows).fill(0);\n        const clusterMap = {};\n        clusters.forEach((cluster, i) => {\n            if (!clusterMap[cluster])\n                clusterMap[cluster] = [];\n            clusterMap[cluster].push(i);\n        });\n        // const nClusters = Object.keys(clusterMap).length;\n        // const perRow = Math.floor(Math.sqrt(nClusters));\n        let clusterNum = 0;\n        const sortedClusterNames = Object.keys(clusterMap);\n        sortedClusterNames.sort((a, b) => clusterMap[b].length - clusterMap[a].length);\n        let perRow = 6;\n        let yOffset = 0;\n        const layoutSize = 5;\n        for (const clusterName of sortedClusterNames) {\n            const cluster = clusterMap[clusterName];\n            const embeddings = await bioLayout(cluster, sparseMatrix, 0.001);\n            if (clusterNum === Math.ceil(perRow / 1.5)) {\n                clusterNum = 0;\n                yOffset += layoutSize / perRow;\n                perRow = Math.ceil(perRow * 1.5);\n            }\n            const offsetX = ((clusterNum % perRow) * layoutSize / perRow) * 1.5;\n            // const offsetY = Math.floor(clusterNum / perRow) * 2;\n            for (let i = 0; i < embeddings.embedX.length; i++) {\n                embedX[cluster[i]] = embeddings.embedX[i] * layoutSize / perRow + offsetX;\n                embedY[cluster[i]] = embeddings.embedY[i] * layoutSize / perRow + yOffset;\n            }\n            clusterNum++;\n        }\n        return { embedX, embedY };\n    }\n    mergeClusters(clusters, i, j) {\n        const iCluster = clusters[i];\n        const jCluster = clusters[j];\n        for (let k = 0; k < clusters.length; k++) {\n            if (clusters[k] === jCluster)\n                clusters[k] = iCluster;\n        }\n    }\n    assignClusters(sparseMatrix, nRows) {\n        let clusterNum = 0;\n        const is = [];\n        const js = [];\n        const order = Math.floor(Math.max(Math.log10(nRows), 2)) + 1;\n        const minOrder = Math.pow(10, order);\n        const clusters = new Array(nRows).fill(-1);\n        for (const i of Object.keys(sparseMatrix)) {\n            for (const j of Object.keys(sparseMatrix[i])) {\n                if (Math.round(sparseMatrix[i][j] * minOrder) / minOrder > 0 &&\n                    sparseMatrix[i][j] !== Number(i) && Number(j) > Number(i)) {\n                    is.push(Number(i));\n                    js.push(Number(j));\n                    if (clusters[Number(i)] !== -1 && clusters[Number(j)] !== -1) {\n                        if (clusters[Number(i)] !== clusters[Number(j)])\n                            this.mergeClusters(clusters, Number(i), Number(j));\n                    }\n                    else if (clusters[Number(i)] !== -1) {\n                        clusters[Number(j)] = clusters[Number(i)];\n                    }\n                    else if (clusters[Number(j)] !== -1) {\n                        clusters[Number(i)] = clusters[Number(j)];\n                    }\n                    else {\n                        clusterNum++;\n                        clusters[Number(i)] = clusterNum;\n                        clusters[Number(j)] = clusterNum;\n                    }\n                }\n            }\n        }\n        for (let i = 0; i < clusters.length; i++) {\n            if (clusters[i] === -1) {\n                clusterNum++;\n                clusters[i] = clusterNum;\n            }\n        }\n        return { clusters, is: new Uint32Array(is), js: new Uint32Array(js) };\n    }\n    /** same as assign clusters but working on the level of CSR format returned from webGPU */\n    assignClustersCSR(mclRes, nRows) {\n        let clusterNum = 0;\n        const is = [];\n        const js = [];\n        const order = Math.floor(Math.max(Math.log10(nRows), 2)) + 1;\n        const minOrder = 1 / Math.pow(10, order);\n        const clusters = new Array(nRows).fill(-1);\n        const correctedOffsets = new Uint32Array(nRows + 1);\n        let offsetCounter = 0;\n        correctedOffsets[0] = 0;\n        for (let i = 0; i < nRows; i++) {\n            for (let k = mclRes.indexOffsets[i]; k < mclRes.indexOffsets[i + 1]; k++) {\n                const j = mclRes.KNNIndexes[k];\n                if (j <= i || mclRes.KNNSimilarities[k] <= minOrder)\n                    continue;\n                is.push(i);\n                js.push(j);\n                offsetCounter++;\n                if (clusters[i] !== -1 && clusters[j] !== -1) {\n                    if (clusters[i] !== clusters[j])\n                        this.mergeClusters(clusters, i, j);\n                }\n                else if (clusters[i] !== -1) {\n                    clusters[j] = clusters[i];\n                }\n                else if (clusters[j] !== -1) {\n                    clusters[i] = clusters[j];\n                }\n                else {\n                    clusterNum++;\n                    clusters[i] = clusterNum;\n                    clusters[j] = clusterNum;\n                }\n            }\n            correctedOffsets[i + 1] = offsetCounter;\n        }\n        for (let i = 0; i < clusters.length; i++) {\n            if (clusters[i] === -1) {\n                clusterNum++;\n                clusters[i] = clusterNum;\n            }\n        }\n        return { clusters, is: new Uint32Array(is), js: new Uint32Array(js), correctedOffsets };\n    }\n    toObjectForm(sparseMatrix) {\n        const sparseObject = {};\n        for (let i = 0; i < sparseMatrix.i.length; i++) {\n            if (!sparseObject[sparseMatrix.i[i]])\n                sparseObject[sparseMatrix.i[i]] = {};\n            sparseObject[sparseMatrix.i[i]][sparseMatrix.j[i]] = 1 - sparseMatrix.distance[i];\n            if (!sparseObject[sparseMatrix.j[i]])\n                sparseObject[sparseMatrix.j[i]] = {};\n            sparseObject[sparseMatrix.j[i]][sparseMatrix.i[i]] = 1 - sparseMatrix.distance[i];\n        }\n        return sparseObject;\n    }\n    addLoops(sparseObject, nRows) {\n        for (let i = 0; i < nRows; i++) {\n            if (!sparseObject[i])\n                sparseObject[i] = {};\n            sparseObject[i][i] = this._options.multFactor;\n        }\n    }\n    normalize(sparseObject) {\n        for (const i of Object.keys(sparseObject)) {\n            const row = sparseObject[i];\n            let sum = 0;\n            for (const j of Object.keys(row))\n                sum += row[j];\n            if (sum === 0)\n                continue;\n            for (const j of Object.keys(row))\n                sparseObject[i][j] /= sum;\n        }\n    }\n    expand(sparseObject, nRows) {\n        const expandedObject = {};\n        const order = Math.floor(Math.max(Math.log10(nRows), 2)) + 1;\n        const minOrder = Math.pow(10, order);\n        for (let i = 0; i < nRows; i++) {\n            if (!sparseObject[i])\n                continue;\n            // const row = sparseObject[i];\n            expandedObject[i] ?? (expandedObject[i] = {});\n            for (let j = i; j < nRows; j++) {\n                if (!sparseObject[i]?.[j])\n                    continue;\n                const val = this.getExpandValue(sparseObject, i, j); //pruning step\n                if (Math.round(val * minOrder) / minOrder > 0) {\n                    expandedObject[i][j] = val;\n                    if (!expandedObject[j])\n                        expandedObject[j] = {};\n                    expandedObject[j][i] = val;\n                }\n            }\n        }\n        return expandedObject;\n    }\n    // private prune(row: SparseMatrixObject[number]) {\n    // }\n    inflate(sparseObject) {\n        for (const i of Object.keys(sparseObject)) {\n            const row = sparseObject[i];\n            for (const j of Object.keys(row))\n                sparseObject[i][j] = Math.pow(sparseObject[i][j], this._options.inflateFactor);\n        }\n    }\n    getExpandValue(sparseObject, i, j) {\n        let val = 0;\n        const currentIndexes = Object.keys(sparseObject[i] ?? {});\n        const otherIndexes = Object.keys(sparseObject[j] ?? {});\n        for (const k of currentIndexes) {\n            if (otherIndexes.includes(k))\n                val += sparseObject[i][k] * sparseObject[j][k];\n        }\n        return val;\n    }\n}\n//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"marcov-cluster.js","sourceRoot":"","sources":["marcov-cluster.ts"],"names":[],"mappings":"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","sourcesContent":["import {SparseMatrixResult} from '../distance-matrix/sparse-matrix-service';\nimport {bioLayout} from './bio-layout';\nimport {MCLOptions, SparseMatrixObject} from './types';\nimport {MCLOpReturnType} from '@datagrok-libraries/math/src/webGPU/MCL/types';\nimport {markovClusterWebGPU} from '@datagrok-libraries/math/src/webGPU/MCL/MCL-webGPU';\nexport const defaultMCLOptions: MCLOptions = {\n  expandFactor: 2,\n  maxIterations: 5,\n  inflateFactor: 2,\n  multFactor: 1,\n};\n\nexport class MCLSparseReducer {\n  private _options: MCLOptions;\n\n  constructor(opts: Partial<MCLOptions> = {}) {\n    this._options = {...defaultMCLOptions, ...opts};\n  }\n\n  public async transform(sparseMatrix: SparseMatrixResult, nRows: number) {\n    // testWorkerMultiply();\n    // return new Int32Array(nRows);\n    let sparseObject = this.toObjectForm(sparseMatrix);\n    if (this._options.maxIterations > 0) {\n      this.addLoops(sparseObject, nRows);\n      this.normalize(sparseObject);\n      for (let i = 0; i < this._options.maxIterations; i++) {\n        sparseObject = this.expand(sparseObject, nRows);\n        this.inflate(sparseObject);\n        this.normalize(sparseObject);\n      }\n    }\n    const {clusters, is, js} = this.assignClusters(sparseObject, nRows);\n    this.correctClusters(clusters);\n    const embeddings = await this.layout(clusters, sparseObject, nRows);\n    return {clusters, embedX: embeddings.embedX, embedY: embeddings.embedY, is, js};\n  }\n\n  public async transformWebGPU(sparseMatrix: SparseMatrixResult, nRows: number) {\n    if (this._options.maxIterations === 0)\n      return this.transform(sparseMatrix, nRows);\n\n    const mclRes = await markovClusterWebGPU(\n      sparseMatrix, nRows, this._options.maxIterations, this._options.inflateFactor\n    );\n    const sparseObject = this.csrToSparseObject(mclRes, nRows);\n    const {clusters, is, js} = this.assignClusters(sparseObject, nRows);\n    this.correctClusters(clusters);\n    const embeddings = await this.layout(clusters, sparseObject, nRows);\n    return {clusters, embedX: embeddings.embedX, embedY: embeddings.embedY, is, js};\n  }\n\n  private correctClusters(clusters: number[]) {\n    const clusterSizeMap: {[_: number]: number} = {};\n    for (const cluster of clusters) {\n      if (!clusterSizeMap[cluster])\n        clusterSizeMap[cluster] = 0;\n      clusterSizeMap[cluster]++;\n    }\n    const sortedIndexes = Object.keys(clusterSizeMap).map(Number).sort((a, b) => clusterSizeMap[b] - clusterSizeMap[a]);\n    const clusterMap: {[_: number]: number} = {};\n    sortedIndexes.forEach((clusterIdx, i) => clusterMap[clusterIdx] = i + 1);\n    for (let i = 0; i < clusters.length; i++)\n      clusters[i] = clusterMap[clusters[i]];\n    // let curCluster = 1;\n    // for (let i = 0; i < clusters.length; i++) {\n    //   if (!clusterMap[clusters[i]]) {\n    //     clusterMap[clusters[i]] = curCluster;\n    //     clusters[i] = curCluster;\n    //     curCluster++;\n    //   } else {\n    //     clusters[i] = clusterMap[clusters[i]];\n    //   }\n    // }\n  }\n\n  private csrToSparseObject(mclRes: MCLOpReturnType, nRows: number) {\n    const order = Math.floor(Math.max(Math.log10(nRows), 2)) + 1;\n    const minOrder = 1 / Math.pow(10, order);\n    const sparseObject: SparseMatrixObject = {};\n    for (let i = 0; i < nRows; i++) {\n      sparseObject[i] = {};\n      for (let k = mclRes.indexOffsets[i]; k < mclRes.indexOffsets[i + 1]; k++) {\n        const j = mclRes.KNNIndexes[k];\n        if (j <= i || mclRes.KNNSimilarities[k] < minOrder)\n          continue;\n        sparseObject[i][j] = mclRes.KNNSimilarities[k];\n      }\n    }\n    return sparseObject;\n  }\n\n  private async layout(clusters: number[], sparseMatrix: SparseMatrixObject, nRows: number) {\n    const embedX = new Float32Array(nRows).fill(0);\n    const embedY = new Float32Array(nRows).fill(0);\n    const clusterMap: {[_: number]: number[]} = {};\n    clusters.forEach((cluster, i) => {\n      if (!clusterMap[cluster])\n        clusterMap[cluster] = [];\n      clusterMap[cluster].push(i);\n    });\n    // const nClusters = Object.keys(clusterMap).length;\n    // const perRow = Math.floor(Math.sqrt(nClusters));\n    let clusterNum = 0;\n    const sortedClusterNames = Object.keys(clusterMap);\n    sortedClusterNames.sort((a, b) => clusterMap[b as any].length - clusterMap[a as any].length);\n    let perRow = 6;\n\n    let yOffset = 0;\n    const layoutSize = 5;\n    for (const clusterName of sortedClusterNames) {\n      const cluster = clusterMap[clusterName as any]!;\n      const embeddings = await bioLayout(cluster, sparseMatrix, 0.001);\n      if (clusterNum === Math.ceil(perRow / 1.5)) {\n        clusterNum = 0;\n        yOffset += layoutSize / perRow;\n        perRow = Math.ceil(perRow * 1.5);\n      }\n      const offsetX = ((clusterNum % perRow) * layoutSize / perRow) * 1.5;\n      // const offsetY = Math.floor(clusterNum / perRow) * 2;\n\n      for (let i = 0; i < embeddings.embedX.length; i++) {\n        embedX[cluster[i]] = embeddings.embedX[i] * layoutSize / perRow + offsetX;\n        embedY[cluster[i]] = embeddings.embedY[i] * layoutSize / perRow + yOffset;\n      }\n      clusterNum++;\n    }\n    return {embedX, embedY};\n  }\n\n  private mergeClusters(clusters: number[], i: number, j: number) {\n    const iCluster = clusters[i];\n    const jCluster = clusters[j];\n    for (let k = 0; k < clusters.length; k++) {\n      if (clusters[k] === jCluster)\n        clusters[k] = iCluster;\n    }\n  }\n  public assignClusters(sparseMatrix: SparseMatrixObject, nRows: number) {\n    let clusterNum = 0;\n    const is: number[] = [];\n    const js: number[] = [];\n    const order = Math.floor(Math.max(Math.log10(nRows), 2)) + 1;\n    const minOrder = Math.pow(10, order);\n    const clusters: number[] = new Array(nRows).fill(-1);\n    for (const i of Object.keys(sparseMatrix)) {\n      for (const j of Object.keys(sparseMatrix[i as any])) {\n        if (Math.round(sparseMatrix[i as any][j as any] * minOrder) / minOrder > 0 &&\n          sparseMatrix[i as any][j as any] !== Number(i) && Number(j) > Number(i)) {\n          is.push(Number(i));\n          js.push(Number(j));\n          if (clusters[Number(i)] !== -1 && clusters[Number(j)] !== -1) {\n            if (clusters[Number(i)] !== clusters[Number(j)])\n              this.mergeClusters(clusters, Number(i), Number(j));\n          } else if (clusters[Number(i)] !== -1) {\n            clusters[Number(j)] = clusters[Number(i)];\n          } else if (clusters[Number(j)] !== -1) {\n            clusters[Number(i)] = clusters[Number(j)];\n          } else {\n            clusterNum++;\n            clusters[Number(i)] = clusterNum;\n            clusters[Number(j)] = clusterNum;\n          }\n        }\n      }\n    }\n    for (let i=0; i < clusters.length; i++) {\n      if (clusters[i] === -1) {\n        clusterNum ++;\n        clusters[i] = clusterNum;\n      }\n    }\n    return {clusters, is: new Uint32Array(is), js: new Uint32Array(js)};\n  }\n\n  /** same as assign clusters but working on the level of CSR format returned from webGPU */\n  private assignClustersCSR(mclRes: MCLOpReturnType, nRows: number) {\n    let clusterNum = 0;\n    const is: number[] = [];\n    const js: number[] = [];\n    const order = Math.floor(Math.max(Math.log10(nRows), 2)) + 1;\n    const minOrder = 1 / Math.pow(10, order);\n    const clusters: number[] = new Array(nRows).fill(-1);\n    const correctedOffsets = new Uint32Array(nRows + 1);\n    let offsetCounter = 0;\n    correctedOffsets[0] = 0;\n    for (let i = 0; i < nRows; i++) {\n      for (let k = mclRes.indexOffsets[i]; k < mclRes.indexOffsets[i + 1]; k++) {\n        const j = mclRes.KNNIndexes[k];\n        if (j <= i || mclRes.KNNSimilarities[k] <= minOrder)\n          continue;\n        is.push(i);\n        js.push(j);\n        offsetCounter++;\n        if (clusters[i] !== -1 && clusters[j] !== -1) {\n          if (clusters[i] !== clusters[j])\n            this.mergeClusters(clusters, i, j);\n        } else if (clusters[i] !== -1) {\n          clusters[j] = clusters[i];\n        } else if (clusters[j] !== -1) {\n          clusters[i] = clusters[j];\n        } else {\n          clusterNum++;\n          clusters[i] = clusterNum;\n          clusters[j] = clusterNum;\n        }\n      }\n      correctedOffsets[i + 1] = offsetCounter;\n    }\n    for (let i=0; i < clusters.length; i++) {\n      if (clusters[i] === -1) {\n        clusterNum ++;\n        clusters[i] = clusterNum;\n      }\n    }\n    return {clusters, is: new Uint32Array(is), js: new Uint32Array(js), correctedOffsets};\n  }\n\n  public toObjectForm(sparseMatrix: SparseMatrixResult): SparseMatrixObject {\n    const sparseObject: {[_: number]: {[_: number]: number}} = {};\n    for (let i = 0; i < sparseMatrix.i.length; i++) {\n      if (!sparseObject[sparseMatrix.i[i]])\n        sparseObject[sparseMatrix.i[i]] = {};\n      sparseObject[sparseMatrix.i[i]][sparseMatrix.j[i]] = 1 - sparseMatrix.distance[i];\n      if (!sparseObject[sparseMatrix.j[i]])\n        sparseObject[sparseMatrix.j[i]] = {};\n      sparseObject[sparseMatrix.j[i]][sparseMatrix.i[i]] = 1 - sparseMatrix.distance[i];\n    }\n    return sparseObject;\n  }\n\n  private addLoops(sparseObject: SparseMatrixObject, nRows: number) {\n    for (let i = 0; i < nRows; i++) {\n      if (!sparseObject[i])\n        sparseObject[i] = {};\n      sparseObject[i][i] = this._options.multFactor;\n    }\n  }\n\n  private normalize(sparseObject: SparseMatrixObject) {\n    for (const i of Object.keys(sparseObject)) {\n      const row = sparseObject[i as any];\n      let sum = 0;\n      for (const j of Object.keys(row))\n        sum += row[j as any];\n      if (sum === 0) continue;\n      for (const j of Object.keys(row))\n        sparseObject[i as any][j as any] /= sum;\n    }\n  }\n\n  private expand(sparseObject: SparseMatrixObject, nRows: number) {\n    const expandedObject: SparseMatrixObject = {};\n    const order = Math.floor(Math.max(Math.log10(nRows), 2)) + 1;\n    const minOrder = Math.pow(10, order);\n    for (let i = 0; i < nRows; i++) {\n      if (!sparseObject[i])\n        continue;\n      // const row = sparseObject[i];\n      expandedObject[i] ??= {};\n      for (let j = i; j < nRows; j++) {\n        if (!sparseObject[i]?.[j])\n          continue;\n        const val = this.getExpandValue(sparseObject, i, j); //pruning step\n        if (Math.round(val * minOrder) / minOrder > 0) {\n          expandedObject[i][j] = val;\n          if (!expandedObject[j])\n            expandedObject[j] = {};\n          expandedObject[j][i] = val;\n        }\n      }\n    }\n    return expandedObject;\n  }\n\n  // private prune(row: SparseMatrixObject[number]) {\n\n  // }\n\n  private inflate(sparseObject: SparseMatrixObject) {\n    for (const i of Object.keys(sparseObject)) {\n      const row = sparseObject[i as any];\n      for (const j of Object.keys(row))\n        sparseObject[i as any][j as any] = Math.pow(sparseObject[i as any][j as any], this._options.inflateFactor);\n    }\n  }\n\n  private getExpandValue(sparseObject: SparseMatrixObject, i: any, j: any) {\n    let val = 0;\n    const currentIndexes = Object.keys(sparseObject[i] ?? {});\n    const otherIndexes = Object.keys(sparseObject[j] ?? {});\n    for (const k of currentIndexes) {\n      if (otherIndexes.includes(k))\n        val += sparseObject[i][k as any] * sparseObject[j][k as any];\n    }\n    return val;\n  }\n}\n\n"]}","var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments || [])).next());\n    });\n};\n/* eslint-disable max-len */\nimport { getGPUDevice } from '../getGPUDevice';\nimport { sparseKNNNoralizeColwise } from './colwise-normalize';\nimport { expandNoRevive, inflate } from './inflate-expand';\nimport { toSparseKNNSimilarityForm } from './utils';\nexport function markovClusterWebGPU(sparseMatrix, nRows, maxIterations = 5, inflateFactor = 2) {\n    return __awaiter(this, void 0, void 0, function* () {\n        const device = yield getGPUDevice();\n        if (!device)\n            throw new Error('no gpu device found');\n        // self loops are already added here\n        const sparseKNNForm = toSparseKNNSimilarityForm(sparseMatrix, nRows);\n        // first, we normilize the similarities\n        yield sparseKNNNoralizeColwise(device, sparseKNNForm.KNNSimilarities, sparseKNNForm.indexOffsets, nRows);\n        let res = sparseKNNForm;\n        // then we expand the similarities\n        for (let i = 0; i < maxIterations; i++) {\n            //console.log(checkSorted(res.KNNIndexes, res.indexOffsets));\n            const expandRes = yield expandNoRevive(device, res.KNNSimilarities, res.KNNIndexes, res.indexOffsets, nRows);\n            // then we inflate the similarities\n            inflate(expandRes.KNNSimilarities, inflateFactor);\n            // then we normilize the similarities again\n            yield sparseKNNNoralizeColwise(device, expandRes.KNNSimilarities, expandRes.indexOffsets, nRows);\n            res = expandRes;\n        }\n        return res;\n    });\n}\n//# sourceMappingURL=MCL-webGPU.js.map","import { multiColWebGPUSparseMatrix } from '@datagrok-libraries/math/src/webGPU/sparse-matrix/webGPU-sparse-matrix';\nimport { SparseMatrixService } from '../distance-matrix/sparse-matrix-service';\nimport { MCLSparseReducer } from './marcov-cluster';\nonmessage = async (event) => {\n    const { data, threshold, weights, aggregationMethod, distanceFnArgs, distanceFns, maxIterations, useWebGPU, inflate } = event.data;\n    console.time('sparse matrix');\n    let sparse = null;\n    if (useWebGPU) {\n        try {\n            sparse = await multiColWebGPUSparseMatrix(data, threshold / 100, distanceFns, aggregationMethod, weights, distanceFnArgs);\n        }\n        catch (e) {\n            console.error(e);\n        }\n    }\n    if (!sparse) { // falsback to CPU\n        if (useWebGPU)\n            console.error('WEBGPU sparse matrix calculation failed, falling back to CPU implementation');\n        sparse = await new SparseMatrixService()\n            .calcMultiColumn(data, distanceFns, threshold / 100, distanceFnArgs, weights, aggregationMethod);\n    }\n    console.timeEnd('sparse matrix');\n    //const res = await new MCLSparseReducer({maxIterations: maxIterations ?? 5}).transform(sparse, data[0].length);\n    const reducer = new MCLSparseReducer({ maxIterations: maxIterations ?? 5, inflateFactor: inflate ?? 2 });\n    console.time('MCL');\n    let res = null;\n    if (useWebGPU) {\n        try {\n            res = await reducer.transformWebGPU(sparse, data[0].length);\n        }\n        catch (e) {\n            console.error('webGPU MCL failed, falling back to CPU implementation');\n            console.error(e);\n        }\n    }\n    if (!res)\n        res = await reducer.transform(sparse, data[0].length);\n    console.timeEnd('MCL');\n    postMessage({ res });\n};\n//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWNsLXdvcmtlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIm1jbC13b3JrZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFDLDBCQUEwQixFQUFDLE1BQU0sd0VBQXdFLENBQUM7QUFDbEgsT0FBTyxFQUFxQixtQkFBbUIsRUFBQyxNQUFNLDBDQUEwQyxDQUFDO0FBR2pHLE9BQU8sRUFBQyxnQkFBZ0IsRUFBQyxNQUFNLGtCQUFrQixDQUFDO0FBRWxELFNBQVMsR0FBRyxLQUFLLEVBQUUsS0FBSyxFQUFFLEVBQUU7SUFDMUIsTUFBTSxFQUFDLElBQUksRUFBRSxTQUFTLEVBQUUsT0FBTyxFQUFFLGlCQUFpQixFQUFFLGNBQWMsRUFBRSxXQUFXLEVBQUUsYUFBYSxFQUFFLFNBQVMsRUFBRSxPQUFPLEVBQUMsR0FNL0csS0FBSyxDQUFDLElBQUksQ0FBQztJQUVmLE9BQU8sQ0FBQyxJQUFJLENBQUMsZUFBZSxDQUFDLENBQUM7SUFDOUIsSUFBSSxNQUFNLEdBQThCLElBQUksQ0FBQztJQUM3QyxJQUFJLFNBQVMsRUFBRSxDQUFDO1FBQ2QsSUFBSSxDQUFDO1lBQ0gsTUFBTSxHQUFHLE1BQU0sMEJBQTBCLENBQ3ZDLElBQUksRUFBRSxTQUFTLEdBQUcsR0FBRyxFQUFFLFdBQWtCLEVBQUUsaUJBQXdCLEVBQUUsT0FBTyxFQUFFLGNBQWMsQ0FBQyxDQUFDO1FBQ2xHLENBQUM7UUFBQyxPQUFPLENBQUMsRUFBRSxDQUFDO1lBQ1gsT0FBTyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNuQixDQUFDO0lBQ0gsQ0FBQztJQUNELElBQUksQ0FBQyxNQUFNLEVBQUUsQ0FBQyxDQUFDLGtCQUFrQjtRQUMvQixJQUFJLFNBQVM7WUFDWCxPQUFPLENBQUMsS0FBSyxDQUFDLDZFQUE2RSxDQUFDLENBQUM7UUFFL0YsTUFBTSxHQUFHLE1BQU0sSUFBSSxtQkFBbUIsRUFBRTthQUNyQyxlQUFlLENBQUMsSUFBSSxFQUFFLFdBQVcsRUFBRSxTQUFTLEdBQUcsR0FBRyxFQUFFLGNBQWMsRUFBRSxPQUFPLEVBQUUsaUJBQWlCLENBQUMsQ0FBQztJQUNyRyxDQUFDO0lBQ0QsT0FBTyxDQUFDLE9BQU8sQ0FBQyxlQUFlLENBQUMsQ0FBQztJQUVqQyxnSEFBZ0g7SUFDaEgsTUFBTSxPQUFPLEdBQUcsSUFBSSxnQkFBZ0IsQ0FBQyxFQUFDLGFBQWEsRUFBRSxhQUFhLElBQUksQ0FBQyxFQUFFLGFBQWEsRUFBRSxPQUFPLElBQUksQ0FBQyxFQUFDLENBQUMsQ0FBQztJQUN2RyxPQUFPLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ3BCLElBQUksR0FBRyxHQUFRLElBQUksQ0FBQztJQUNwQixJQUFJLFNBQVMsRUFBRSxDQUFDO1FBQ2QsSUFBSSxDQUFDO1lBQ0gsR0FBRyxHQUFHLE1BQU0sT0FBTyxDQUFDLGVBQWUsQ0FBQyxNQUFNLEVBQUUsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBQzlELENBQUM7UUFBQyxPQUFPLENBQUMsRUFBRSxDQUFDO1lBQ1gsT0FBTyxDQUFDLEtBQUssQ0FBQyx1REFBdUQsQ0FBQyxDQUFDO1lBQ3ZFLE9BQU8sQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDbkIsQ0FBQztJQUNILENBQUM7SUFDRCxJQUFJLENBQUMsR0FBRztRQUNOLEdBQUcsR0FBRyxNQUFNLE9BQU8sQ0FBQyxTQUFTLENBQUMsTUFBTSxFQUFFLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUN4RCxPQUFPLENBQUMsT0FBTyxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBRXZCLFdBQVcsQ0FBQyxFQUFDLEdBQUcsRUFBQyxDQUFDLENBQUM7QUFDckIsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHttdWx0aUNvbFdlYkdQVVNwYXJzZU1hdHJpeH0gZnJvbSAnQGRhdGFncm9rLWxpYnJhcmllcy9tYXRoL3NyYy93ZWJHUFUvc3BhcnNlLW1hdHJpeC93ZWJHUFUtc3BhcnNlLW1hdHJpeCc7XG5pbXBvcnQge1NwYXJzZU1hdHJpeFJlc3VsdCwgU3BhcnNlTWF0cml4U2VydmljZX0gZnJvbSAnLi4vZGlzdGFuY2UtbWF0cml4L3NwYXJzZS1tYXRyaXgtc2VydmljZSc7XG5pbXBvcnQge0Rpc3RhbmNlQWdncmVnYXRpb25NZXRob2R9IGZyb20gJy4uL2Rpc3RhbmNlLW1hdHJpeC90eXBlcyc7XG5pbXBvcnQge0tub3duTWV0cmljc30gZnJvbSAnLi4vdHlwZWQtbWV0cmljcyc7XG5pbXBvcnQge01DTFNwYXJzZVJlZHVjZXJ9IGZyb20gJy4vbWFyY292LWNsdXN0ZXInO1xuXG5vbm1lc3NhZ2UgPSBhc3luYyAoZXZlbnQpID0+IHtcbiAgY29uc3Qge2RhdGEsIHRocmVzaG9sZCwgd2VpZ2h0cywgYWdncmVnYXRpb25NZXRob2QsIGRpc3RhbmNlRm5BcmdzLCBkaXN0YW5jZUZucywgbWF4SXRlcmF0aW9ucywgdXNlV2ViR1BVLCBpbmZsYXRlfTpcbiAgIHtcbiAgICBkYXRhOiBhbnlbXVtdLCB0aHJlc2hvbGQ6IG51bWJlcixcbiAgICB3ZWlnaHRzOiBudW1iZXJbXSwgYWdncmVnYXRpb25NZXRob2Q6IERpc3RhbmNlQWdncmVnYXRpb25NZXRob2QsXG4gICAgZGlzdGFuY2VGbnM6IEtub3duTWV0cmljc1tdLCBkaXN0YW5jZUZuQXJnczogYW55W10sIG1heEl0ZXJhdGlvbnM6IG51bWJlcixcbiAgICB1c2VXZWJHUFU/OiBib29sZWFuLCBpbmZsYXRlPzogbnVtYmVyXG4gIH0gPSBldmVudC5kYXRhO1xuXG4gIGNvbnNvbGUudGltZSgnc3BhcnNlIG1hdHJpeCcpO1xuICBsZXQgc3BhcnNlOiBTcGFyc2VNYXRyaXhSZXN1bHQgfCBudWxsID0gbnVsbDtcbiAgaWYgKHVzZVdlYkdQVSkge1xuICAgIHRyeSB7XG4gICAgICBzcGFyc2UgPSBhd2FpdCBtdWx0aUNvbFdlYkdQVVNwYXJzZU1hdHJpeChcbiAgICAgICAgZGF0YSwgdGhyZXNob2xkIC8gMTAwLCBkaXN0YW5jZUZucyBhcyBhbnksIGFnZ3JlZ2F0aW9uTWV0aG9kIGFzIGFueSwgd2VpZ2h0cywgZGlzdGFuY2VGbkFyZ3MpO1xuICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgIGNvbnNvbGUuZXJyb3IoZSk7XG4gICAgfVxuICB9XG4gIGlmICghc3BhcnNlKSB7IC8vIGZhbHNiYWNrIHRvIENQVVxuICAgIGlmICh1c2VXZWJHUFUpXG4gICAgICBjb25zb2xlLmVycm9yKCdXRUJHUFUgc3BhcnNlIG1hdHJpeCBjYWxjdWxhdGlvbiBmYWlsZWQsIGZhbGxpbmcgYmFjayB0byBDUFUgaW1wbGVtZW50YXRpb24nKTtcblxuICAgIHNwYXJzZSA9IGF3YWl0IG5ldyBTcGFyc2VNYXRyaXhTZXJ2aWNlKClcbiAgICAgIC5jYWxjTXVsdGlDb2x1bW4oZGF0YSwgZGlzdGFuY2VGbnMsIHRocmVzaG9sZCAvIDEwMCwgZGlzdGFuY2VGbkFyZ3MsIHdlaWdodHMsIGFnZ3JlZ2F0aW9uTWV0aG9kKTtcbiAgfVxuICBjb25zb2xlLnRpbWVFbmQoJ3NwYXJzZSBtYXRyaXgnKTtcblxuICAvL2NvbnN0IHJlcyA9IGF3YWl0IG5ldyBNQ0xTcGFyc2VSZWR1Y2VyKHttYXhJdGVyYXRpb25zOiBtYXhJdGVyYXRpb25zID8/IDV9KS50cmFuc2Zvcm0oc3BhcnNlLCBkYXRhWzBdLmxlbmd0aCk7XG4gIGNvbnN0IHJlZHVjZXIgPSBuZXcgTUNMU3BhcnNlUmVkdWNlcih7bWF4SXRlcmF0aW9uczogbWF4SXRlcmF0aW9ucyA/PyA1LCBpbmZsYXRlRmFjdG9yOiBpbmZsYXRlID8/IDJ9KTtcbiAgY29uc29sZS50aW1lKCdNQ0wnKTtcbiAgbGV0IHJlczogYW55ID0gbnVsbDtcbiAgaWYgKHVzZVdlYkdQVSkge1xuICAgIHRyeSB7XG4gICAgICByZXMgPSBhd2FpdCByZWR1Y2VyLnRyYW5zZm9ybVdlYkdQVShzcGFyc2UsIGRhdGFbMF0ubGVuZ3RoKTtcbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICBjb25zb2xlLmVycm9yKCd3ZWJHUFUgTUNMIGZhaWxlZCwgZmFsbGluZyBiYWNrIHRvIENQVSBpbXBsZW1lbnRhdGlvbicpO1xuICAgICAgY29uc29sZS5lcnJvcihlKTtcbiAgICB9XG4gIH1cbiAgaWYgKCFyZXMpXG4gICAgcmVzID0gYXdhaXQgcmVkdWNlci50cmFuc2Zvcm0oc3BhcnNlLCBkYXRhWzBdLmxlbmd0aCk7XG4gIGNvbnNvbGUudGltZUVuZCgnTUNMJyk7XG5cbiAgcG9zdE1lc3NhZ2Uoe3Jlc30pO1xufTtcbiJdfQ=="],"names":["WEBGSLAGGREGATION","__webpack_require__","chunkId","g","globalThis","this","Function","e","window","o","obj","prop","Object","prototype","hasOwnProperty","call","scriptUrl","importScripts","location","document","currentScript","src","scripts","getElementsByTagName","length","i","test","Error","replace","p","b","self","WEBGSLAGGREGATIONFUNCTIONS","EUCLIDEAN","arraySize","MANHATTAN","WEBGPUDISTANCE","webGPUFunctions","HAMMING","_maxArraySize","entryIndex","maxArraySize","_entryIndex","TANIMOTO","LEVENSTEIN","NEEDLEMAN_WUNSCH","MONOMER_CHEMICAL_DISTANCE","SOKAL","COSINE","ASYMMETRIC","Difference","OneHot","distanceFunctionComplexity","maxEntrySize","Math","ceil","_maxEntrySize","TypeSupportedDistances","Set","__awaiter","thisArg","_arguments","P","generator","Promise","resolve","reject","fulfilled","value","step","next","rejected","result","done","then","apply","gpuAdapter","gpuDevice","getGPUDevice","navigator","gpu","requestAdapter","powerPreference","isLost","lost","r","setTimeout","requiredBufferSize","adapterLimits","limits","buffferSizeLimit","maxBufferSize","storageBufferSizeLimit","maxStorageBufferBindingSize","requestDevice","requiredLimits","min","console","error","DistanceAggregationMethods","multiColWebGPUSparseMatrix","entryList","threshold","distanceMetrics","aggregationFunction","weights","options","device","availableDistanceMetrics","values","some","metric","includes","join","maxDistance","list","numOfColumns","listSize","processInfo","map","entry","distanceMetric","gapOpenPenalty","gapExtensionPenalty","_a","_b","entryType","encodedList","Uint32Array","split","c","charCodeAt","Float32Array","_data","Int32Array","encodedListType","arraySizes","arr","has","maxEntryLen","reduce","a","max","complexity","EncodedArrayConstructor","flatSourceArray","forEach","seq","set","suppInfoStructWgsl","suppInfoSize","suppInfoType","suppInfoBuffer","maxMonomerIndex","scoringMatrix","alphabetIndexes","keys","prev","n","Array","fill","String","fromCharCode","similarityMatrixSize","transferedSimilarityMatrix","key","key2","offset","range","dataTypeWGSL","dataStructWgsl","sourceArraySize","webGPUProcessInfo","suppInfoWgsl","info","filter","wgsl","needsDummy","trim","dataWgsl","numOfThreads","sparseResultSizePerThread","combinedComplexity","maxIterationsPerThread","workgroupsDim","sqrt","globalThreadDimSize","condensedDistanceMatrixSize","dmChunkSizePerThread","module","createShaderModule","label","code","getCombinedDistanceScript","pipeline","createComputePipeline","layout","compute","entryPoint","startAtCols","startAtRows","endAtCols","endAtRows","chunkSize","floor","startRow","startCol","time","endIdx","endRow","endCol","timeEnd","computeInfoBuffer32Size","suppInfoBuffer32Size","sparseMatrixEachArray32Size","computeInfoBufferSize","BYTES_PER_ELEMENT","paddedComputeInfoBufferSize","remainder","computeInfoBuffer","createBuffer","size","usage","GPUBufferUsage","STORAGE","COPY_SRC","COPY_DST","mappedAtCreation","mappedComputeInfoArrayBuffer","getMappedRange","computeInfoOffSet","ArrayConstructor","unmap","suppInfoBufferSize","paddedSuppInfoBufferSize","suppInfoRemainder","mappedSuppInfoArrayBuffer","suppInfoOffSet","byteLength","resultsBufferSize","paddedResultsBufferSize","resultsRemainder","resultsBuffer","bindGroup","createBindGroup","getBindGroupLayout","entries","binding","resource","buffer","resultsOutBuffer","MAP_READ","resultIs","resultJs","resultDistances","isAllDone","encoder","createCommandEncoder","pass","beginComputePass","setPipeline","setBindGroup","dispatchWorkgroups","end","copyBufferToBuffer","commandBuffer","finish","queue","submit","onSubmittedWorkDone","mapAsync","GPUMapMode","READ","resultsOutArrayBuffer","resultOffset","resultsI","resultsJ","resultsDistances","resultsFound","every","d","totalResults","combinedI","combinedJ","combinedDistances","combinedOffset","resI","found","subarray","push","totalSize","finalI","finalJ","finalDistances","finalOffset","destroy","j","distance","maxEntryLens","aggregation","_","isNil","x","insertSmaller","distancesAr","indexes","num","index","newPosition","findIndex","v","pop","splice","SparseMatrixService","constructor","_workerCount","hardwareConcurrency","calcMultiColumn","fnNames","opts","aggregationMethod","matSize","minThreshold","getMinimalThreshold","log","promises","workers","Worker","URL","idx","resolveWorker","rejectWorker","startIdx","postMessage","onmessage","data","terminate","results","all","fullSize","acc","val","res","calc","fnName","getKNN","nNeighbours","multiColumnKNN","getThresholdKNN","multiColumnThresholdKnn","knnDistances","knnIndexes","knnSizes","knnRes","getSampleDistances","thresholdWorkers","maxSampleSize","sampleSise","testSetSizePerWorker","tPromises","sampleLength","sort","w","calcSync","distanceFn","distances","cnt","mi","mj","async","bioLayout","cluster","sparseObject","is","js","embedX","random","embedY","xVelocities","yVelocities","temperature","dx","dy","abs","vecSize","minX","minY","maxX","maxY","rangeX","rangeY","getLayoutEmbeddings","sparseKNNNoralizeColwise","knnSimilarities","offsets","nRows","neededThreads","neededWorkGroups","workGroupThreadsPerDim","workGroupDim","threadsPerDim","simmilaritiesBuffer","offsetsBuffer","commandEncoder","passEncoder","outSimilarityBuffer","arrayBuffer","inflate","factor","pow","expandNoRevive","order","log10","pruneValue","outKNNSimilarities","sparseKNNBufferByteSize","sparseKNNBuffer","sparseKNNArrayBuffer","rowIndexes","getRowIndexes","startBuffer","resultBlockBuffer","outBlockBuffer","start","curLength","writeBuffer","outBlock","KNNIndexes","KNNSimilarities","indexOffsets","toSparseKNNSimilarityForm","sparseMatrix","occurenceCount","countOccurancesAddLoops","offsetForm","ar","insertCounter","row","col","similarity","toString","defaultMCLOptions","expandFactor","maxIterations","inflateFactor","multFactor","MCLSparseReducer","_options","transform","toObjectForm","addLoops","normalize","expand","clusters","assignClusters","correctClusters","embeddings","transformWebGPU","mclRes","sparseKNNForm","expandRes","markovClusterWebGPU","csrToSparseObject","clusterSizeMap","sortedIndexes","Number","clusterMap","clusterIdx","minOrder","k","clusterNum","sortedClusterNames","perRow","yOffset","clusterName","offsetX","mergeClusters","iCluster","jCluster","round","assignClustersCSR","correctedOffsets","offsetCounter","sum","expandedObject","getExpandValue","currentIndexes","otherIndexes","event","distanceFnArgs","distanceFns","useWebGPU","sparse","reducer"],"sourceRoot":""}
|
|
2
|
+
//# sourceMappingURL=795.js.map
|