@jbrowse/plugin-spreadsheet-view 2.6.1
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/LICENSE +201 -0
- package/dist/LaunchSpreadsheetView/index.d.ts +3 -0
- package/dist/LaunchSpreadsheetView/index.js +26 -0
- package/dist/LaunchSpreadsheetView/index.js.map +1 -0
- package/dist/SpreadsheetView/components/CellData.d.ts +9 -0
- package/dist/SpreadsheetView/components/CellData.js +16 -0
- package/dist/SpreadsheetView/components/CellData.js.map +1 -0
- package/dist/SpreadsheetView/components/ColumnFilterControls.d.ts +3 -0
- package/dist/SpreadsheetView/components/ColumnFilterControls.js +62 -0
- package/dist/SpreadsheetView/components/ColumnFilterControls.js.map +1 -0
- package/dist/SpreadsheetView/components/ColumnMenu.d.ts +16 -0
- package/dist/SpreadsheetView/components/ColumnMenu.js +127 -0
- package/dist/SpreadsheetView/components/ColumnMenu.js.map +1 -0
- package/dist/SpreadsheetView/components/DataRow.d.ts +12 -0
- package/dist/SpreadsheetView/components/DataRow.js +82 -0
- package/dist/SpreadsheetView/components/DataRow.js.map +1 -0
- package/dist/SpreadsheetView/components/DataTable.d.ts +10 -0
- package/dist/SpreadsheetView/components/DataTable.js +43 -0
- package/dist/SpreadsheetView/components/DataTable.js.map +1 -0
- package/dist/SpreadsheetView/components/DataTableHeader.d.ts +6 -0
- package/dist/SpreadsheetView/components/DataTableHeader.js +102 -0
- package/dist/SpreadsheetView/components/DataTableHeader.js.map +1 -0
- package/dist/SpreadsheetView/components/GlobalFilterControls.d.ts +5 -0
- package/dist/SpreadsheetView/components/GlobalFilterControls.js +64 -0
- package/dist/SpreadsheetView/components/GlobalFilterControls.js.map +1 -0
- package/dist/SpreadsheetView/components/ImportWizard.d.ts +6 -0
- package/dist/SpreadsheetView/components/ImportWizard.js +81 -0
- package/dist/SpreadsheetView/components/ImportWizard.js.map +1 -0
- package/dist/SpreadsheetView/components/NumberEditor.d.ts +9 -0
- package/dist/SpreadsheetView/components/NumberEditor.js +54 -0
- package/dist/SpreadsheetView/components/NumberEditor.js.map +1 -0
- package/dist/SpreadsheetView/components/RowCountMessage.d.ts +8 -0
- package/dist/SpreadsheetView/components/RowCountMessage.js +33 -0
- package/dist/SpreadsheetView/components/RowCountMessage.js.map +1 -0
- package/dist/SpreadsheetView/components/RowMenu.d.ts +9 -0
- package/dist/SpreadsheetView/components/RowMenu.js +39 -0
- package/dist/SpreadsheetView/components/RowMenu.js.map +1 -0
- package/dist/SpreadsheetView/components/SortIndicator.d.ts +6 -0
- package/dist/SpreadsheetView/components/SortIndicator.js +28 -0
- package/dist/SpreadsheetView/components/SortIndicator.js.map +1 -0
- package/dist/SpreadsheetView/components/Spreadsheet.d.ts +11 -0
- package/dist/SpreadsheetView/components/Spreadsheet.js +25 -0
- package/dist/SpreadsheetView/components/Spreadsheet.js.map +1 -0
- package/dist/SpreadsheetView/components/SpreadsheetView.d.ts +6 -0
- package/dist/SpreadsheetView/components/SpreadsheetView.js +86 -0
- package/dist/SpreadsheetView/components/SpreadsheetView.js.map +1 -0
- package/dist/SpreadsheetView/components/StatusBar.d.ts +11 -0
- package/dist/SpreadsheetView/components/StatusBar.js +42 -0
- package/dist/SpreadsheetView/components/StatusBar.js.map +1 -0
- package/dist/SpreadsheetView/components/util.d.ts +1 -0
- package/dist/SpreadsheetView/components/util.js +19 -0
- package/dist/SpreadsheetView/components/util.js.map +1 -0
- package/dist/SpreadsheetView/importAdapters/BedImport.d.ts +17 -0
- package/dist/SpreadsheetView/importAdapters/BedImport.js +147 -0
- package/dist/SpreadsheetView/importAdapters/BedImport.js.map +1 -0
- package/dist/SpreadsheetView/importAdapters/ImportUtils.d.ts +42 -0
- package/dist/SpreadsheetView/importAdapters/ImportUtils.js +124 -0
- package/dist/SpreadsheetView/importAdapters/ImportUtils.js.map +1 -0
- package/dist/SpreadsheetView/importAdapters/STARFusionImport.d.ts +9 -0
- package/dist/SpreadsheetView/importAdapters/STARFusionImport.js +73 -0
- package/dist/SpreadsheetView/importAdapters/STARFusionImport.js.map +1 -0
- package/dist/SpreadsheetView/importAdapters/VcfImport.d.ts +13 -0
- package/dist/SpreadsheetView/importAdapters/VcfImport.js +111 -0
- package/dist/SpreadsheetView/importAdapters/VcfImport.js.map +1 -0
- package/dist/SpreadsheetView/index.d.ts +4 -0
- package/dist/SpreadsheetView/index.js +42 -0
- package/dist/SpreadsheetView/index.js.map +1 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/LocEnd.d.ts +13 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/LocEnd.js +21 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/LocEnd.js.map +1 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/LocRef.d.ts +13 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/LocRef.js +21 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/LocRef.js.map +1 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/LocStart.d.ts +13 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/LocStart.js +21 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/LocStart.js.map +1 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/LocString.d.ts +13 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/LocString.js +190 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/LocString.js.map +1 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/MakeSpreadsheetColumnType.d.ts +19 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/MakeSpreadsheetColumnType.js +24 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/MakeSpreadsheetColumnType.js.map +1 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/Number.d.ts +30 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/Number.js +130 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/Number.js.map +1 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/Text.d.ts +28 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/Text.js +131 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/Text.js.map +1 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/index.d.ts +72 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/index.js +38 -0
- package/dist/SpreadsheetView/models/ColumnDataTypes/index.js.map +1 -0
- package/dist/SpreadsheetView/models/FilterControls.d.ts +26 -0
- package/dist/SpreadsheetView/models/FilterControls.js +88 -0
- package/dist/SpreadsheetView/models/FilterControls.js.map +1 -0
- package/dist/SpreadsheetView/models/ImportWizard.d.ts +31 -0
- package/dist/SpreadsheetView/models/ImportWizard.js +193 -0
- package/dist/SpreadsheetView/models/ImportWizard.js.map +1 -0
- package/dist/SpreadsheetView/models/Row.d.ts +17 -0
- package/dist/SpreadsheetView/models/Row.js +46 -0
- package/dist/SpreadsheetView/models/Row.js.map +1 -0
- package/dist/SpreadsheetView/models/Spreadsheet.d.ts +256 -0
- package/dist/SpreadsheetView/models/Spreadsheet.js +176 -0
- package/dist/SpreadsheetView/models/Spreadsheet.js.map +1 -0
- package/dist/SpreadsheetView/models/SpreadsheetView.d.ts +485 -0
- package/dist/SpreadsheetView/models/SpreadsheetView.js +204 -0
- package/dist/SpreadsheetView/models/SpreadsheetView.js.map +1 -0
- package/dist/SpreadsheetView/models/StaticRowSet.d.ts +163 -0
- package/dist/SpreadsheetView/models/StaticRowSet.js +60 -0
- package/dist/SpreadsheetView/models/StaticRowSet.js.map +1 -0
- package/dist/index.d.ts +8 -0
- package/dist/index.js +36 -0
- package/dist/index.js.map +1 -0
- package/esm/LaunchSpreadsheetView/index.d.ts +3 -0
- package/esm/LaunchSpreadsheetView/index.js +24 -0
- package/esm/LaunchSpreadsheetView/index.js.map +1 -0
- package/esm/SpreadsheetView/components/CellData.d.ts +9 -0
- package/esm/SpreadsheetView/components/CellData.js +11 -0
- package/esm/SpreadsheetView/components/CellData.js.map +1 -0
- package/esm/SpreadsheetView/components/ColumnFilterControls.d.ts +3 -0
- package/esm/SpreadsheetView/components/ColumnFilterControls.js +57 -0
- package/esm/SpreadsheetView/components/ColumnFilterControls.js.map +1 -0
- package/esm/SpreadsheetView/components/ColumnMenu.d.ts +16 -0
- package/esm/SpreadsheetView/components/ColumnMenu.js +122 -0
- package/esm/SpreadsheetView/components/ColumnMenu.js.map +1 -0
- package/esm/SpreadsheetView/components/DataRow.d.ts +12 -0
- package/esm/SpreadsheetView/components/DataRow.js +77 -0
- package/esm/SpreadsheetView/components/DataRow.js.map +1 -0
- package/esm/SpreadsheetView/components/DataTable.d.ts +10 -0
- package/esm/SpreadsheetView/components/DataTable.js +38 -0
- package/esm/SpreadsheetView/components/DataTable.js.map +1 -0
- package/esm/SpreadsheetView/components/DataTableHeader.d.ts +6 -0
- package/esm/SpreadsheetView/components/DataTableHeader.js +74 -0
- package/esm/SpreadsheetView/components/DataTableHeader.js.map +1 -0
- package/esm/SpreadsheetView/components/GlobalFilterControls.d.ts +5 -0
- package/esm/SpreadsheetView/components/GlobalFilterControls.js +36 -0
- package/esm/SpreadsheetView/components/GlobalFilterControls.js.map +1 -0
- package/esm/SpreadsheetView/components/ImportWizard.d.ts +6 -0
- package/esm/SpreadsheetView/components/ImportWizard.js +53 -0
- package/esm/SpreadsheetView/components/ImportWizard.js.map +1 -0
- package/esm/SpreadsheetView/components/NumberEditor.d.ts +9 -0
- package/esm/SpreadsheetView/components/NumberEditor.js +29 -0
- package/esm/SpreadsheetView/components/NumberEditor.js.map +1 -0
- package/esm/SpreadsheetView/components/RowCountMessage.d.ts +8 -0
- package/esm/SpreadsheetView/components/RowCountMessage.js +28 -0
- package/esm/SpreadsheetView/components/RowCountMessage.js.map +1 -0
- package/esm/SpreadsheetView/components/RowMenu.d.ts +9 -0
- package/esm/SpreadsheetView/components/RowMenu.js +34 -0
- package/esm/SpreadsheetView/components/RowMenu.js.map +1 -0
- package/esm/SpreadsheetView/components/SortIndicator.d.ts +6 -0
- package/esm/SpreadsheetView/components/SortIndicator.js +22 -0
- package/esm/SpreadsheetView/components/SortIndicator.js.map +1 -0
- package/esm/SpreadsheetView/components/Spreadsheet.d.ts +11 -0
- package/esm/SpreadsheetView/components/Spreadsheet.js +20 -0
- package/esm/SpreadsheetView/components/Spreadsheet.js.map +1 -0
- package/esm/SpreadsheetView/components/SpreadsheetView.d.ts +6 -0
- package/esm/SpreadsheetView/components/SpreadsheetView.js +58 -0
- package/esm/SpreadsheetView/components/SpreadsheetView.js.map +1 -0
- package/esm/SpreadsheetView/components/StatusBar.d.ts +11 -0
- package/esm/SpreadsheetView/components/StatusBar.js +37 -0
- package/esm/SpreadsheetView/components/StatusBar.js.map +1 -0
- package/esm/SpreadsheetView/components/util.d.ts +1 -0
- package/esm/SpreadsheetView/components/util.js +15 -0
- package/esm/SpreadsheetView/components/util.js.map +1 -0
- package/esm/SpreadsheetView/importAdapters/BedImport.d.ts +17 -0
- package/esm/SpreadsheetView/importAdapters/BedImport.js +141 -0
- package/esm/SpreadsheetView/importAdapters/BedImport.js.map +1 -0
- package/esm/SpreadsheetView/importAdapters/ImportUtils.d.ts +42 -0
- package/esm/SpreadsheetView/importAdapters/ImportUtils.js +95 -0
- package/esm/SpreadsheetView/importAdapters/ImportUtils.js.map +1 -0
- package/esm/SpreadsheetView/importAdapters/STARFusionImport.d.ts +9 -0
- package/esm/SpreadsheetView/importAdapters/STARFusionImport.js +69 -0
- package/esm/SpreadsheetView/importAdapters/STARFusionImport.js.map +1 -0
- package/esm/SpreadsheetView/importAdapters/VcfImport.d.ts +13 -0
- package/esm/SpreadsheetView/importAdapters/VcfImport.js +103 -0
- package/esm/SpreadsheetView/importAdapters/VcfImport.js.map +1 -0
- package/esm/SpreadsheetView/index.d.ts +4 -0
- package/esm/SpreadsheetView/index.js +14 -0
- package/esm/SpreadsheetView/index.js.map +1 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/LocEnd.d.ts +13 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/LocEnd.js +16 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/LocEnd.js.map +1 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/LocRef.d.ts +13 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/LocRef.js +16 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/LocRef.js.map +1 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/LocStart.d.ts +13 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/LocStart.js +16 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/LocStart.js.map +1 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/LocString.d.ts +13 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/LocString.js +185 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/LocString.js.map +1 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/MakeSpreadsheetColumnType.d.ts +19 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/MakeSpreadsheetColumnType.js +21 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/MakeSpreadsheetColumnType.js.map +1 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/Number.d.ts +30 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/Number.js +123 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/Number.js.map +1 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/Text.d.ts +28 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/Text.js +124 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/Text.js.map +1 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/index.d.ts +72 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/index.js +30 -0
- package/esm/SpreadsheetView/models/ColumnDataTypes/index.js.map +1 -0
- package/esm/SpreadsheetView/models/FilterControls.d.ts +26 -0
- package/esm/SpreadsheetView/models/FilterControls.js +86 -0
- package/esm/SpreadsheetView/models/FilterControls.js.map +1 -0
- package/esm/SpreadsheetView/models/ImportWizard.d.ts +31 -0
- package/esm/SpreadsheetView/models/ImportWizard.js +168 -0
- package/esm/SpreadsheetView/models/ImportWizard.js.map +1 -0
- package/esm/SpreadsheetView/models/Row.d.ts +17 -0
- package/esm/SpreadsheetView/models/Row.js +44 -0
- package/esm/SpreadsheetView/models/Row.js.map +1 -0
- package/esm/SpreadsheetView/models/Spreadsheet.d.ts +256 -0
- package/esm/SpreadsheetView/models/Spreadsheet.js +171 -0
- package/esm/SpreadsheetView/models/Spreadsheet.js.map +1 -0
- package/esm/SpreadsheetView/models/SpreadsheetView.d.ts +485 -0
- package/esm/SpreadsheetView/models/SpreadsheetView.js +199 -0
- package/esm/SpreadsheetView/models/SpreadsheetView.js.map +1 -0
- package/esm/SpreadsheetView/models/StaticRowSet.d.ts +163 -0
- package/esm/SpreadsheetView/models/StaticRowSet.js +55 -0
- package/esm/SpreadsheetView/models/StaticRowSet.js.map +1 -0
- package/esm/index.d.ts +8 -0
- package/esm/index.js +30 -0
- package/esm/index.js.map +1 -0
- package/package.json +65 -0
- package/src/LaunchSpreadsheetView/index.ts +40 -0
- package/src/SpreadsheetView/components/CellData.tsx +35 -0
- package/src/SpreadsheetView/components/ColumnFilterControls.tsx +84 -0
- package/src/SpreadsheetView/components/ColumnMenu.tsx +166 -0
- package/src/SpreadsheetView/components/DataRow.tsx +126 -0
- package/src/SpreadsheetView/components/DataTable.tsx +91 -0
- package/src/SpreadsheetView/components/DataTableHeader.tsx +122 -0
- package/src/SpreadsheetView/components/GlobalFilterControls.tsx +70 -0
- package/src/SpreadsheetView/components/ImportWizard.tsx +139 -0
- package/src/SpreadsheetView/components/NumberEditor.tsx +50 -0
- package/src/SpreadsheetView/components/RowCountMessage.tsx +44 -0
- package/src/SpreadsheetView/components/RowMenu.tsx +65 -0
- package/src/SpreadsheetView/components/SortIndicator.tsx +36 -0
- package/src/SpreadsheetView/components/Spreadsheet.test.ts +17 -0
- package/src/SpreadsheetView/components/Spreadsheet.tsx +46 -0
- package/src/SpreadsheetView/components/SpreadsheetView.tsx +119 -0
- package/src/SpreadsheetView/components/StatusBar.tsx +75 -0
- package/src/SpreadsheetView/components/util.ts +16 -0
- package/src/SpreadsheetView/importAdapters/BedImport.test.ts +14 -0
- package/src/SpreadsheetView/importAdapters/BedImport.ts +155 -0
- package/src/SpreadsheetView/importAdapters/ImportUtils.test.ts +25 -0
- package/src/SpreadsheetView/importAdapters/ImportUtils.ts +152 -0
- package/src/SpreadsheetView/importAdapters/STARFusionImport.test.ts +27 -0
- package/src/SpreadsheetView/importAdapters/STARFusionImport.ts +74 -0
- package/src/SpreadsheetView/importAdapters/VcfImport.test.ts +42 -0
- package/src/SpreadsheetView/importAdapters/VcfImport.ts +118 -0
- package/src/SpreadsheetView/importAdapters/__snapshots__/ImportUtils.test.ts.snap +6555 -0
- package/src/SpreadsheetView/importAdapters/__snapshots__/STARFusionImport.test.ts.snap +2354 -0
- package/src/SpreadsheetView/importAdapters/__snapshots__/VcfImport.test.ts.snap +16329 -0
- package/src/SpreadsheetView/index.ts +20 -0
- package/src/SpreadsheetView/models/ColumnDataTypes/LocEnd.ts +21 -0
- package/src/SpreadsheetView/models/ColumnDataTypes/LocRef.ts +21 -0
- package/src/SpreadsheetView/models/ColumnDataTypes/LocStart.ts +21 -0
- package/src/SpreadsheetView/models/ColumnDataTypes/LocString.tsx +328 -0
- package/src/SpreadsheetView/models/ColumnDataTypes/MakeSpreadsheetColumnType.tsx +38 -0
- package/src/SpreadsheetView/models/ColumnDataTypes/Number.tsx +177 -0
- package/src/SpreadsheetView/models/ColumnDataTypes/Text.tsx +173 -0
- package/src/SpreadsheetView/models/ColumnDataTypes/index.ts +34 -0
- package/src/SpreadsheetView/models/FilterControls.ts +95 -0
- package/src/SpreadsheetView/models/ImportWizard.ts +203 -0
- package/src/SpreadsheetView/models/Row.ts +46 -0
- package/src/SpreadsheetView/models/Spreadsheet.ts +209 -0
- package/src/SpreadsheetView/models/SpreadsheetView.test.ts +8 -0
- package/src/SpreadsheetView/models/SpreadsheetView.ts +243 -0
- package/src/SpreadsheetView/models/StaticRowSet.ts +65 -0
- package/src/SpreadsheetView/test_data/1801160099-N32519_26611_S51_56704.hard-filtered.vcf +174 -0
- package/src/SpreadsheetView/test_data/breast_cancer.subset.csv +50 -0
- package/src/SpreadsheetView/test_data/diabetes_target.csv.gz +0 -0
- package/src/SpreadsheetView/test_data/foo.bed +11 -0
- package/src/SpreadsheetView/test_data/location_test.csv +3 -0
- package/src/SpreadsheetView/test_data/location_test_2.csv +3 -0
- package/src/SpreadsheetView/test_data/reads_lr_skbr3.fa_ngmlr-0.2.3_mapped.bam.sniffles1kb_auto_l8_s5_noalt.bedpe +1405 -0
- package/src/SpreadsheetView/test_data/starfusion_example.fusion_predictions.tsv +25 -0
- package/src/SpreadsheetView/test_data/volvox.sort.bed +109 -0
- package/src/index.ts +38 -0
|
@@ -0,0 +1,155 @@
|
|
|
1
|
+
import { ParseOptions, parseTsvBuffer } from './ImportUtils'
|
|
2
|
+
|
|
3
|
+
const browserBytes = 'browser '.split('').map(c => c.charCodeAt(0))
|
|
4
|
+
const trackBytes = 'track '.split('').map(c => c.charCodeAt(0))
|
|
5
|
+
const commentBytes = '#'.split('').map(c => c.charCodeAt(0))
|
|
6
|
+
|
|
7
|
+
function bytesAreFoundAt(position: number, buffer: Buffer, bytes: number[]) {
|
|
8
|
+
let i = 0
|
|
9
|
+
for (; i < bytes.length; i += 1) {
|
|
10
|
+
if (buffer[position + i] !== bytes[i]) {
|
|
11
|
+
return false
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
return true
|
|
15
|
+
}
|
|
16
|
+
export function removeBedHeaders(buffer: Buffer) {
|
|
17
|
+
// slice off the first lines of the buffer if it starts with one or more
|
|
18
|
+
// header lines
|
|
19
|
+
let i = 0
|
|
20
|
+
for (; i < buffer.length; i += 1) {
|
|
21
|
+
if (
|
|
22
|
+
bytesAreFoundAt(i, buffer, browserBytes) ||
|
|
23
|
+
bytesAreFoundAt(i, buffer, trackBytes) ||
|
|
24
|
+
bytesAreFoundAt(i, buffer, commentBytes)
|
|
25
|
+
) {
|
|
26
|
+
// consume up to the next newline
|
|
27
|
+
do {
|
|
28
|
+
i += 1
|
|
29
|
+
} while (buffer[i] !== 10)
|
|
30
|
+
} else {
|
|
31
|
+
// end of headers, return
|
|
32
|
+
break
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
if (i) {
|
|
36
|
+
return buffer.slice(i)
|
|
37
|
+
}
|
|
38
|
+
return buffer
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
export async function parseBedBuffer(buffer: Buffer, options: ParseOptions) {
|
|
42
|
+
const b = removeBedHeaders(buffer)
|
|
43
|
+
const data = await parseTsvBuffer(b)
|
|
44
|
+
const bedColumns = [
|
|
45
|
+
{ name: 'chrom', dataType: { type: 'LocRef' } },
|
|
46
|
+
{ name: 'chromStart', dataType: { type: 'LocStart' } },
|
|
47
|
+
{ name: 'chromEnd', dataType: { type: 'LocEnd' } },
|
|
48
|
+
{ name: 'name', dataType: { type: 'Text' } },
|
|
49
|
+
{ name: 'score', dataType: { type: 'Number' } },
|
|
50
|
+
{ name: 'strand', dataType: { type: 'Text' } },
|
|
51
|
+
]
|
|
52
|
+
data.columns.forEach((col, colNumber) => {
|
|
53
|
+
const bedColumn = bedColumns[colNumber]
|
|
54
|
+
if (bedColumn) {
|
|
55
|
+
col.name = bedColumn.name
|
|
56
|
+
col.dataType = bedColumn.dataType
|
|
57
|
+
}
|
|
58
|
+
})
|
|
59
|
+
data.hasColumnNames = true
|
|
60
|
+
data.assemblyName = options.selectedAssemblyName
|
|
61
|
+
|
|
62
|
+
data.columnDisplayOrder.push(data.columnDisplayOrder.length)
|
|
63
|
+
data.columns.unshift({
|
|
64
|
+
name: 'Location',
|
|
65
|
+
dataType: { type: 'LocString' },
|
|
66
|
+
isDerived: true,
|
|
67
|
+
derivationFunctionText: `jexl:{text:row.cells[0].text+':'+row.cells[1].text+'..'+row.cells[2].text,\n
|
|
68
|
+
extendedData: {refName: row.cells.ref.text, start: parseInt(row.cells.start.text,10), end: parseInt(row.cells.end.text,10)}}`,
|
|
69
|
+
})
|
|
70
|
+
return data
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
export async function parseBedPEBuffer(buffer: Buffer, options: ParseOptions) {
|
|
74
|
+
const b = removeBedHeaders(buffer)
|
|
75
|
+
const data = await parseTsvBuffer(b)
|
|
76
|
+
interface BedColumn {
|
|
77
|
+
name: string
|
|
78
|
+
dataType: {
|
|
79
|
+
type: string
|
|
80
|
+
}
|
|
81
|
+
featureField: string[]
|
|
82
|
+
}
|
|
83
|
+
const bedColumns: BedColumn[] = [
|
|
84
|
+
{ name: 'chrom1', dataType: { type: 'Text' }, featureField: ['refName'] },
|
|
85
|
+
{ name: 'start1', dataType: { type: 'Number' }, featureField: ['start'] },
|
|
86
|
+
{ name: 'end1', dataType: { type: 'Number' }, featureField: ['end'] },
|
|
87
|
+
{
|
|
88
|
+
name: 'chrom2',
|
|
89
|
+
dataType: { type: 'Text' },
|
|
90
|
+
featureField: ['mate', 'refName'],
|
|
91
|
+
},
|
|
92
|
+
{
|
|
93
|
+
name: 'start2',
|
|
94
|
+
dataType: { type: 'Number' },
|
|
95
|
+
featureField: ['mate', 'start'],
|
|
96
|
+
},
|
|
97
|
+
{
|
|
98
|
+
name: 'end2',
|
|
99
|
+
dataType: { type: 'Number' },
|
|
100
|
+
featureField: ['mate', 'end'],
|
|
101
|
+
},
|
|
102
|
+
{ name: 'name', dataType: { type: 'Text' }, featureField: ['name'] },
|
|
103
|
+
{ name: 'score', dataType: { type: 'Number' }, featureField: ['score'] },
|
|
104
|
+
{ name: 'strand1', dataType: { type: 'Text' }, featureField: ['strand'] },
|
|
105
|
+
{
|
|
106
|
+
name: 'strand2',
|
|
107
|
+
dataType: { type: 'Text' },
|
|
108
|
+
featureField: ['mate', 'strand'],
|
|
109
|
+
},
|
|
110
|
+
]
|
|
111
|
+
data.columns.forEach((col, colNumber) => {
|
|
112
|
+
const bedColumn = bedColumns[colNumber]
|
|
113
|
+
if (bedColumn) {
|
|
114
|
+
col.name = bedColumn.name
|
|
115
|
+
col.dataType = bedColumn.dataType
|
|
116
|
+
}
|
|
117
|
+
})
|
|
118
|
+
data.hasColumnNames = true
|
|
119
|
+
|
|
120
|
+
// decorate each row with a feature object in its extendedData
|
|
121
|
+
data.rowSet.rows.forEach((row, rowNumber) => {
|
|
122
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
123
|
+
const featureData: Record<string, any> = {}
|
|
124
|
+
row.cells.forEach(({ text }, columnNumber) => {
|
|
125
|
+
const bedColumn = bedColumns[columnNumber]
|
|
126
|
+
const val =
|
|
127
|
+
bedColumn && bedColumn.dataType.type === 'Number' && text
|
|
128
|
+
? parseFloat(text)
|
|
129
|
+
: text
|
|
130
|
+
if (bedColumn) {
|
|
131
|
+
// a predefined column
|
|
132
|
+
if (bedColumn.featureField.length === 2) {
|
|
133
|
+
if (!featureData[bedColumn.featureField[0]]) {
|
|
134
|
+
featureData[bedColumn.featureField[0]] = {}
|
|
135
|
+
}
|
|
136
|
+
featureData[bedColumn.featureField[0]][bedColumn.featureField[1]] =
|
|
137
|
+
val
|
|
138
|
+
} else {
|
|
139
|
+
featureData[bedColumn.featureField[0]] = val
|
|
140
|
+
}
|
|
141
|
+
} else {
|
|
142
|
+
// some other column
|
|
143
|
+
featureData[`column${columnNumber + 1}`] = val
|
|
144
|
+
}
|
|
145
|
+
})
|
|
146
|
+
featureData.uniqueId = `bedpe-${rowNumber}`
|
|
147
|
+
row.extendedData = {
|
|
148
|
+
feature: featureData,
|
|
149
|
+
}
|
|
150
|
+
})
|
|
151
|
+
|
|
152
|
+
data.assemblyName = options.selectedAssemblyName
|
|
153
|
+
|
|
154
|
+
return data
|
|
155
|
+
}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { promises as fsPromises } from 'fs'
|
|
2
|
+
import path from 'path'
|
|
3
|
+
|
|
4
|
+
// locals
|
|
5
|
+
import { parseCsvBuffer } from './ImportUtils'
|
|
6
|
+
import SpreadsheetModel from '../models/Spreadsheet'
|
|
7
|
+
|
|
8
|
+
test('csv to spreadsheet snapshot', async () => {
|
|
9
|
+
const filepath = path.join(
|
|
10
|
+
__dirname,
|
|
11
|
+
'..',
|
|
12
|
+
'test_data',
|
|
13
|
+
'breast_cancer.subset.csv',
|
|
14
|
+
)
|
|
15
|
+
const buf = await fsPromises.readFile(filepath)
|
|
16
|
+
const spreadsheetSnap = await parseCsvBuffer(buf, {
|
|
17
|
+
hasColumnNameLine: true,
|
|
18
|
+
columnNameLineNumber: 1,
|
|
19
|
+
isValidRefName: () => true,
|
|
20
|
+
})
|
|
21
|
+
expect(spreadsheetSnap).toMatchSnapshot()
|
|
22
|
+
// @ts-expect-error
|
|
23
|
+
const spreadsheet = SpreadsheetModel.create(spreadsheetSnap)
|
|
24
|
+
expect(spreadsheet.rowSet.rows.length).toBe(49)
|
|
25
|
+
})
|
|
@@ -0,0 +1,152 @@
|
|
|
1
|
+
import { parseLocString } from '@jbrowse/core/util'
|
|
2
|
+
|
|
3
|
+
export function bufferToString(buffer: Buffer) {
|
|
4
|
+
return new TextDecoder('utf8', { fatal: true }).decode(buffer)
|
|
5
|
+
}
|
|
6
|
+
|
|
7
|
+
async function parseWith(buffer: Buffer, options = {}) {
|
|
8
|
+
const csv = await import('csvtojson').then(module => module.default)
|
|
9
|
+
return csv({ noheader: true, output: 'csv', ...options }).fromString(
|
|
10
|
+
bufferToString(buffer),
|
|
11
|
+
)
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export interface Row {
|
|
15
|
+
id: string
|
|
16
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
17
|
+
extendedData?: any
|
|
18
|
+
cells: {
|
|
19
|
+
text: string
|
|
20
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
21
|
+
extendedData?: any
|
|
22
|
+
}[]
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
export interface RowSet {
|
|
26
|
+
isLoaded: boolean
|
|
27
|
+
rows: Row[]
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export interface ParseOptions {
|
|
31
|
+
hasColumnNameLine?: boolean
|
|
32
|
+
columnNameLineNumber?: number
|
|
33
|
+
selectedAssemblyName?: string
|
|
34
|
+
isValidRefName?: (refName: string, assemblyName?: string) => boolean
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export interface Column {
|
|
38
|
+
name: string
|
|
39
|
+
dataType: { type: string }
|
|
40
|
+
isDerived?: boolean
|
|
41
|
+
derivationFunctionText?: string
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
function guessColumnType(
|
|
45
|
+
rowSet: RowSet,
|
|
46
|
+
columnNumber: number,
|
|
47
|
+
isValidRefName: (refName: string, assemblyName?: string) => boolean,
|
|
48
|
+
) {
|
|
49
|
+
const text = rowSet.rows[0].cells[columnNumber].text || ''
|
|
50
|
+
|
|
51
|
+
let guessedType = 'Text'
|
|
52
|
+
|
|
53
|
+
let parsedLoc
|
|
54
|
+
try {
|
|
55
|
+
parsedLoc = parseLocString(text, isValidRefName)
|
|
56
|
+
} catch (error) {
|
|
57
|
+
//
|
|
58
|
+
}
|
|
59
|
+
if (parsedLoc && parsedLoc.refName && typeof parsedLoc.start === 'number') {
|
|
60
|
+
guessedType = 'LocString'
|
|
61
|
+
} else if (/^\d+(\.\d+)?$/.test(text)) {
|
|
62
|
+
guessedType = 'Number'
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
// MAYBE TODO: iterate over the rest of the rows to confirm
|
|
66
|
+
// the type for all the rows
|
|
67
|
+
|
|
68
|
+
return guessedType
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
function dataToSpreadsheetSnapshot(
|
|
72
|
+
rows: string[][],
|
|
73
|
+
options: ParseOptions = {},
|
|
74
|
+
) {
|
|
75
|
+
const {
|
|
76
|
+
hasColumnNameLine = false,
|
|
77
|
+
columnNameLineNumber = 1,
|
|
78
|
+
isValidRefName = () => false,
|
|
79
|
+
selectedAssemblyName,
|
|
80
|
+
} = options
|
|
81
|
+
// rows is an array of row objects and columnNames
|
|
82
|
+
// is an array of column names (in import order)
|
|
83
|
+
let maxCols = 0
|
|
84
|
+
const rowSet: RowSet = {
|
|
85
|
+
isLoaded: true,
|
|
86
|
+
rows: rows.map((row, rowNumber) => {
|
|
87
|
+
const id = rowNumber + (hasColumnNameLine ? 0 : 1)
|
|
88
|
+
if (row.length > maxCols) {
|
|
89
|
+
maxCols = row.length
|
|
90
|
+
}
|
|
91
|
+
return {
|
|
92
|
+
id: String(id),
|
|
93
|
+
cells: row.map((text, columnNumber) => {
|
|
94
|
+
return { columnNumber, text }
|
|
95
|
+
}),
|
|
96
|
+
}
|
|
97
|
+
}),
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
// process the column names row if present
|
|
101
|
+
const columnNames: Record<string, string> = {}
|
|
102
|
+
if (hasColumnNameLine && columnNameLineNumber !== undefined) {
|
|
103
|
+
const [colNamesRow] = rowSet.rows.splice(columnNameLineNumber - 1, 1)
|
|
104
|
+
|
|
105
|
+
if (colNamesRow) {
|
|
106
|
+
colNamesRow.cells.forEach((cell, columnNumber) => {
|
|
107
|
+
columnNames[columnNumber] = cell.text || ''
|
|
108
|
+
})
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
// make our column definitions
|
|
113
|
+
const columns: Column[] = []
|
|
114
|
+
const columnDisplayOrder = []
|
|
115
|
+
for (let columnNumber = 0; columnNumber < maxCols; columnNumber += 1) {
|
|
116
|
+
columnDisplayOrder.push(columnNumber)
|
|
117
|
+
const guessedType = guessColumnType(rowSet, columnNumber, isValidRefName)
|
|
118
|
+
|
|
119
|
+
// store extendeddata for LocString column
|
|
120
|
+
if (guessedType === 'LocString') {
|
|
121
|
+
for (const row of rowSet.rows) {
|
|
122
|
+
const cell = row.cells[columnNumber]
|
|
123
|
+
cell.extendedData = parseLocString(cell.text, isValidRefName)
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
columns[columnNumber] = {
|
|
128
|
+
name: columnNames[columnNumber],
|
|
129
|
+
dataType: {
|
|
130
|
+
type: guessedType,
|
|
131
|
+
},
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
return {
|
|
136
|
+
rowSet,
|
|
137
|
+
columnDisplayOrder,
|
|
138
|
+
hasColumnNames: !!hasColumnNameLine,
|
|
139
|
+
columns,
|
|
140
|
+
assemblyName: selectedAssemblyName,
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
export async function parseCsvBuffer(buffer: Buffer, options?: ParseOptions) {
|
|
145
|
+
const rows = await parseWith(buffer)
|
|
146
|
+
return dataToSpreadsheetSnapshot(rows, options)
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
export async function parseTsvBuffer(buffer: Buffer, options?: ParseOptions) {
|
|
150
|
+
const rows = await parseWith(buffer, { delimiter: '\t' })
|
|
151
|
+
return dataToSpreadsheetSnapshot(rows, options)
|
|
152
|
+
}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { promises as fsPromises } from 'fs'
|
|
2
|
+
import path from 'path'
|
|
3
|
+
|
|
4
|
+
// locals
|
|
5
|
+
import { parseSTARFusionBuffer } from './STARFusionImport'
|
|
6
|
+
import SpreadsheetModel from '../models/Spreadsheet'
|
|
7
|
+
|
|
8
|
+
test('starfusion import', async () => {
|
|
9
|
+
const filepath = path.join(
|
|
10
|
+
__dirname,
|
|
11
|
+
'..',
|
|
12
|
+
'test_data',
|
|
13
|
+
'starfusion_example.fusion_predictions.tsv',
|
|
14
|
+
)
|
|
15
|
+
const buf = await fsPromises.readFile(filepath)
|
|
16
|
+
const spreadsheetSnap = await parseSTARFusionBuffer(buf, {
|
|
17
|
+
selectedAssemblyName: 'fogbat',
|
|
18
|
+
isValidRefName() {
|
|
19
|
+
return true
|
|
20
|
+
},
|
|
21
|
+
})
|
|
22
|
+
expect(spreadsheetSnap).toMatchSnapshot()
|
|
23
|
+
|
|
24
|
+
// @ts-expect-error
|
|
25
|
+
const spreadsheet = SpreadsheetModel.create(spreadsheetSnap)
|
|
26
|
+
expect(spreadsheet.rowSet.rows.length).toBe(24)
|
|
27
|
+
})
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import { parseTsvBuffer, ParseOptions } from './ImportUtils'
|
|
2
|
+
|
|
3
|
+
function parseSTARFusionBreakpointString(str: string) {
|
|
4
|
+
const fields = str.split(':')
|
|
5
|
+
const refName = fields[0]
|
|
6
|
+
const pos = Number.parseInt(fields[1], 10)
|
|
7
|
+
const strand = fields[2] === '-' ? -1 : 1
|
|
8
|
+
return { refName, pos, strand }
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
const numericColumns: Record<string, boolean> = {
|
|
12
|
+
SpanningFragCount: true,
|
|
13
|
+
FFPM: true,
|
|
14
|
+
LeftBreakEntropy: true,
|
|
15
|
+
RightBreakEntropy: true,
|
|
16
|
+
JunctionReadCount: true,
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export async function parseSTARFusionBuffer(
|
|
20
|
+
buffer: Buffer,
|
|
21
|
+
options: ParseOptions,
|
|
22
|
+
) {
|
|
23
|
+
const data = await parseTsvBuffer(buffer, {
|
|
24
|
+
hasColumnNameLine: true,
|
|
25
|
+
columnNameLineNumber: 1,
|
|
26
|
+
selectedAssemblyName: options.selectedAssemblyName,
|
|
27
|
+
isValidRefName: () => false,
|
|
28
|
+
})
|
|
29
|
+
|
|
30
|
+
// remove the # in #FusionName
|
|
31
|
+
data.columns[0].name = data.columns[0].name.replace('#', '')
|
|
32
|
+
// set some columns to be numeric
|
|
33
|
+
data.columns.forEach(col => {
|
|
34
|
+
if (numericColumns[col.name]) {
|
|
35
|
+
col.dataType = { type: 'Number' }
|
|
36
|
+
}
|
|
37
|
+
})
|
|
38
|
+
|
|
39
|
+
// decorate each row with a feature object in its extendedData
|
|
40
|
+
data.rowSet.rows.forEach((row, rowNumber) => {
|
|
41
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
42
|
+
const featureData: Record<string, any> = {}
|
|
43
|
+
row.cells.forEach(({ text }, columnNumber) => {
|
|
44
|
+
const column = data.columns[columnNumber]
|
|
45
|
+
if (column.name === 'LeftBreakpoint' && text) {
|
|
46
|
+
const { refName, pos, strand } = parseSTARFusionBreakpointString(text)
|
|
47
|
+
featureData.refName = refName
|
|
48
|
+
featureData.start = pos
|
|
49
|
+
featureData.end = pos
|
|
50
|
+
featureData.strand = strand
|
|
51
|
+
} else if (column.name === 'RightBreakpoint' && text) {
|
|
52
|
+
const { refName, pos, strand } = parseSTARFusionBreakpointString(text)
|
|
53
|
+
featureData.mate = {
|
|
54
|
+
refName,
|
|
55
|
+
start: pos,
|
|
56
|
+
end: pos,
|
|
57
|
+
strand,
|
|
58
|
+
}
|
|
59
|
+
} else if (text && numericColumns[column.name]) {
|
|
60
|
+
// some other column, numeric
|
|
61
|
+
featureData[column.name] = parseFloat(text)
|
|
62
|
+
} else {
|
|
63
|
+
// some other column, text
|
|
64
|
+
featureData[column.name] = text
|
|
65
|
+
}
|
|
66
|
+
})
|
|
67
|
+
featureData.uniqueId = `sf-${rowNumber + 1}`
|
|
68
|
+
row.extendedData = {
|
|
69
|
+
feature: featureData,
|
|
70
|
+
}
|
|
71
|
+
})
|
|
72
|
+
|
|
73
|
+
return data
|
|
74
|
+
}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { promises as fsPromises } from 'fs'
|
|
2
|
+
import path from 'path'
|
|
3
|
+
import { parseVcfBuffer, splitVcfFileHeaderAndBody } from './VcfImport'
|
|
4
|
+
import SpreadsheetModel from '../models/Spreadsheet'
|
|
5
|
+
|
|
6
|
+
describe('vcf file splitter', () => {
|
|
7
|
+
const cases: [string, {}][] = [
|
|
8
|
+
[
|
|
9
|
+
'##fileformat=VCFv4.3\nfogbat\n',
|
|
10
|
+
{ header: '##fileformat=VCFv4.3\n', body: 'fogbat\n' },
|
|
11
|
+
],
|
|
12
|
+
[
|
|
13
|
+
'##fileformat=VCFv4.3\n##zonker\n##deek\n##donk\nfogbat\n',
|
|
14
|
+
{
|
|
15
|
+
header: '##fileformat=VCFv4.3\n##zonker\n##deek\n##donk\n',
|
|
16
|
+
body: 'fogbat\n',
|
|
17
|
+
},
|
|
18
|
+
],
|
|
19
|
+
]
|
|
20
|
+
|
|
21
|
+
cases.forEach(([input, output], caseNumber) => {
|
|
22
|
+
test(`case ${caseNumber}`, () => {
|
|
23
|
+
expect(splitVcfFileHeaderAndBody(input)).toEqual(output)
|
|
24
|
+
})
|
|
25
|
+
})
|
|
26
|
+
})
|
|
27
|
+
|
|
28
|
+
test('vcf file import', async () => {
|
|
29
|
+
const filepath = path.join(
|
|
30
|
+
__dirname,
|
|
31
|
+
'..',
|
|
32
|
+
'test_data',
|
|
33
|
+
'1801160099-N32519_26611_S51_56704.hard-filtered.vcf',
|
|
34
|
+
)
|
|
35
|
+
const buf = await fsPromises.readFile(filepath)
|
|
36
|
+
const spreadsheetSnap = parseVcfBuffer(buf)
|
|
37
|
+
expect(spreadsheetSnap).toMatchSnapshot()
|
|
38
|
+
|
|
39
|
+
// @ts-expect-error
|
|
40
|
+
const spreadsheet = SpreadsheetModel.create(spreadsheetSnap)
|
|
41
|
+
expect(spreadsheet.rowSet.rows.length).toBe(101)
|
|
42
|
+
})
|
|
@@ -0,0 +1,118 @@
|
|
|
1
|
+
import VCF from '@gmod/vcf'
|
|
2
|
+
import { VcfFeature } from '@jbrowse/plugin-variants'
|
|
3
|
+
import {
|
|
4
|
+
bufferToString,
|
|
5
|
+
Row,
|
|
6
|
+
RowSet,
|
|
7
|
+
Column,
|
|
8
|
+
ParseOptions,
|
|
9
|
+
} from './ImportUtils'
|
|
10
|
+
|
|
11
|
+
const vcfCoreColumns: { name: string; type: string }[] = [
|
|
12
|
+
{ name: 'CHROM', type: 'Text' }, // 0
|
|
13
|
+
{ name: 'POS', type: 'Number' }, // 1
|
|
14
|
+
{ name: 'ID', type: 'Text' }, // 2
|
|
15
|
+
{ name: 'REF', type: 'Text' }, // 3
|
|
16
|
+
{ name: 'ALT', type: 'Text' }, // 4
|
|
17
|
+
{ name: 'QUAL', type: 'Number' }, // 5
|
|
18
|
+
{ name: 'FILTER', type: 'Text' }, // 6
|
|
19
|
+
{ name: 'INFO', type: 'Text' }, // 7
|
|
20
|
+
{ name: 'FORMAT', type: 'Text' }, // 8
|
|
21
|
+
]
|
|
22
|
+
|
|
23
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
24
|
+
function vcfRecordToRow(vcfParser: any, line: string, lineNumber: number): Row {
|
|
25
|
+
const vcfVariant = vcfParser.parseLine(line)
|
|
26
|
+
const vcfFeature = new VcfFeature({
|
|
27
|
+
variant: vcfVariant,
|
|
28
|
+
parser: vcfParser,
|
|
29
|
+
id: `vcf-${lineNumber}`,
|
|
30
|
+
})
|
|
31
|
+
|
|
32
|
+
const data = line.split('\t').map(d => (d === '.' ? '' : d))
|
|
33
|
+
// no format column, add blank
|
|
34
|
+
if (data.length === 8) {
|
|
35
|
+
data.push('')
|
|
36
|
+
}
|
|
37
|
+
const row: Row = {
|
|
38
|
+
id: String(lineNumber + 1),
|
|
39
|
+
extendedData: { vcfFeature: vcfFeature.toJSON() },
|
|
40
|
+
cells: data.map((text, columnNumber) => {
|
|
41
|
+
return {
|
|
42
|
+
columnNumber,
|
|
43
|
+
text,
|
|
44
|
+
}
|
|
45
|
+
}),
|
|
46
|
+
}
|
|
47
|
+
return row
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
export function parseVcfBuffer(buffer: Buffer, options: ParseOptions = {}) {
|
|
51
|
+
const { selectedAssemblyName } = options
|
|
52
|
+
let { header, body } = splitVcfFileHeaderAndBody(bufferToString(buffer))
|
|
53
|
+
const rows: Row[] = []
|
|
54
|
+
const vcfParser = new VCF({ header })
|
|
55
|
+
header = '' // garbage collect
|
|
56
|
+
body.split(/\n|\r\n|\r/).forEach((line: string, lineNumber) => {
|
|
57
|
+
if (/\S/.test(line)) {
|
|
58
|
+
rows.push(vcfRecordToRow(vcfParser, line, lineNumber))
|
|
59
|
+
}
|
|
60
|
+
})
|
|
61
|
+
body = '' // garbage collect
|
|
62
|
+
|
|
63
|
+
const rowSet: RowSet = {
|
|
64
|
+
isLoaded: true,
|
|
65
|
+
rows,
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
const columnDisplayOrder: number[] = []
|
|
69
|
+
const columns: Column[] = []
|
|
70
|
+
for (let i = 0; i < vcfCoreColumns.length; i += 1) {
|
|
71
|
+
columnDisplayOrder.push(i)
|
|
72
|
+
columns[i] = {
|
|
73
|
+
name: vcfCoreColumns[i].name,
|
|
74
|
+
dataType: { type: vcfCoreColumns[i].type },
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
for (let i = 0; i < vcfParser.samples.length; i += 1) {
|
|
78
|
+
const oi = vcfCoreColumns.length + i
|
|
79
|
+
columnDisplayOrder.push(oi)
|
|
80
|
+
columns[oi] = { name: vcfParser.samples[i], dataType: { type: 'Text' } }
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
columnDisplayOrder.push(columnDisplayOrder.length)
|
|
84
|
+
columns.unshift({
|
|
85
|
+
name: 'Location',
|
|
86
|
+
dataType: { type: 'LocString' },
|
|
87
|
+
isDerived: true,
|
|
88
|
+
derivationFunctionText: `jexl:{text:row.extendedData.vcfFeature.refName+':'\n
|
|
89
|
+
+row.extendedData.vcfFeature.start+'..'+row.extendedData.vcfFeature.end, extendedData:\n
|
|
90
|
+
{refName:row.extendedData.vcfFeature.refName,start:row.extendedData.vcfFeature.start,end:row.extendedData.vcfFeature.end}}`,
|
|
91
|
+
})
|
|
92
|
+
|
|
93
|
+
return {
|
|
94
|
+
rowSet,
|
|
95
|
+
columnDisplayOrder,
|
|
96
|
+
hasColumnNames: true,
|
|
97
|
+
columns,
|
|
98
|
+
assemblyName: selectedAssemblyName,
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
export function splitVcfFileHeaderAndBody(wholeFile: string) {
|
|
103
|
+
// split into header and the rest of the file
|
|
104
|
+
let headerEndIndex = 0
|
|
105
|
+
let prevChar
|
|
106
|
+
for (; headerEndIndex < wholeFile.length; headerEndIndex += 1) {
|
|
107
|
+
const c = wholeFile[headerEndIndex]
|
|
108
|
+
if (prevChar === '\n' && c !== '#') {
|
|
109
|
+
break
|
|
110
|
+
}
|
|
111
|
+
prevChar = c
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
return {
|
|
115
|
+
header: wholeFile.slice(0, Math.max(0, headerEndIndex)),
|
|
116
|
+
body: wholeFile.slice(headerEndIndex),
|
|
117
|
+
}
|
|
118
|
+
}
|