helpers.ts 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. /**
  2. * Copyright (c) 2023 mol* contributors, licensed under MIT, See LICENSE file for more info.
  3. *
  4. * @author Sebastian Bittrich <sebastian.bittrich@rcsb.org>
  5. */
  6. import { Vec3 } from 'molstar/lib/mol-math/linear-algebra/3d/vec3';
  7. import { Structure } from 'molstar/lib/mol-model/structure/structure/structure';
  8. import { PluginStateObject } from 'molstar/lib/mol-plugin-state/objects';
  9. import { determineBackboneAtom, MAX_EXCHANGES, MAX_MOTIF_SIZE } from './validation';
  10. import { StructureElement, StructureProperties, to_mmCIF } from 'molstar/lib/mol-model/structure/structure';
  11. import { StructureSelectionHistoryEntry } from 'molstar/lib/mol-plugin-state/manager/structure/selection';
  12. import { Residue } from '../strucmotif';
  13. import { PluginContext } from 'molstar/lib/mol-plugin/context';
  14. export type ExchangeState = number;
  15. export type ResidueSelection = { label_asym_id: string, struct_oper_id: string, label_seq_id: number }
  16. export type Exchange = { residue_id: ResidueSelection, allowed: string[] }
  17. const STATIC_URL_REGEX = /^https?:\/\/(models|files).rcsb.org\//;
  18. const FILE_STORAGE_URL = 'https://teilen-dev.rcsb.org/v1/';
  19. const FILE_STORAGE_PUT_URL = FILE_STORAGE_URL + 'putMultipart';
  20. const FILE_STORAGE_GET_URL = FILE_STORAGE_URL + 'download/';
  21. const location = StructureElement.Location.create(void 0);
  22. export function createCtx(plugin: PluginContext, structure: Structure, residueMap: Map<StructureSelectionHistoryEntry, Residue>) {
  23. return {
  24. plugin,
  25. structure,
  26. entryId: structure.model.entryId,
  27. pdbId: new Set<string>(),
  28. sg: new Set<number>(),
  29. hkl: new Set<string>(),
  30. ncs: new Set<number>(),
  31. residueIds: new Array<ResidueSelection>(),
  32. residueMap,
  33. exchanges: new Array<Exchange>(),
  34. coordinates: new Array<{ coords: Vec3, residueId: ResidueSelection }>(),
  35. dataSource: void 0,
  36. format: void 0,
  37. url: void 0
  38. };
  39. }
  40. export type StrucmotifCtx = ReturnType<typeof createCtx>;
  41. export function detectDataSource(ctx: StrucmotifCtx) {
  42. const { plugin, structure } = ctx;
  43. const parent = plugin.helpers.substructureParent.get(structure)!;
  44. const dataCell = plugin.state.data.selectQ(q => q.byValue(parent).rootOfType([PluginStateObject.Data.Binary, PluginStateObject.Data.Blob, PluginStateObject.Data.String]))[0];
  45. const url = dataCell.params?.values.url?.url || dataCell.params?.values.url; // nested is the Import UI component, flat is via method call
  46. const format = PluginStateObject.Data.Binary.is(dataCell.obj) ? 'bcif' :
  47. !!plugin.state.data.selectQ(q => q.byValue(parent).rootOfType(PluginStateObject.Format.Cif))[0] ? 'cif' : 'pdb';
  48. if (!url) {
  49. Object.assign(ctx, { dataSource: 'file', url, format });
  50. } else {
  51. Object.assign(ctx, { dataSource: STATIC_URL_REGEX.test(url) ? 'identifier' : 'url', url, format });
  52. }
  53. }
  54. export function extractResidues(ctx: StrucmotifCtx, loci: StructureSelectionHistoryEntry[]) {
  55. const { x, y, z } = StructureProperties.atom;
  56. for (let i = 0; i < Math.min(MAX_MOTIF_SIZE, loci.length); i++) {
  57. const l = loci[i];
  58. const { structure, elements } = l.loci;
  59. // only first element and only first index will be considered (ignoring multiple residues)
  60. if (!determineBackboneAtom(structure, location, elements[0])) {
  61. alert(`No CA or C4' atom for selected residue`);
  62. return;
  63. }
  64. ctx.pdbId.add(structure.model.entryId);
  65. ctx.sg.add(StructureProperties.unit.spgrOp(location));
  66. ctx.hkl.add(StructureProperties.unit.hkl(location).join('-'));
  67. ctx.ncs.add(StructureProperties.unit.struct_ncs_oper_id(location));
  68. const struct_oper_list_ids = StructureProperties.unit.pdbx_struct_oper_list_ids(location);
  69. const struct_oper_id = join(struct_oper_list_ids);
  70. // handle pure residue-info
  71. const residueId = {
  72. label_asym_id: StructureProperties.chain.label_asym_id(location),
  73. // can be empty array if model is selected
  74. struct_oper_id,
  75. label_seq_id: StructureProperties.residue.label_seq_id(location)
  76. };
  77. ctx.residueIds.push(residueId);
  78. // retrieve CA/C4', used to compute residue distance
  79. const coords = [x(location), y(location), z(location)] as Vec3;
  80. ctx.coordinates.push({ coords, residueId });
  81. // handle potential exchanges - can be empty if deselected by users
  82. const residueMapEntry = ctx.residueMap.get(l)!;
  83. if (residueMapEntry.exchanges?.size > 0) {
  84. if (residueMapEntry.exchanges.size > MAX_EXCHANGES) {
  85. alert(`Maximum number of exchanges per position is ${MAX_EXCHANGES} - Please remove some exchanges from residue ${residueId.label_seq_id} | ${residueId.label_asym_id} | ${residueId.struct_oper_id}.`);
  86. return;
  87. }
  88. ctx.exchanges.push({ residue_id: residueId, allowed: Array.from(residueMapEntry.exchanges.values()) });
  89. }
  90. }
  91. }
  92. function join(opers: any[]) {
  93. // this makes the assumptions that '1' is the identity operator
  94. if (!opers || !opers.length) return '1';
  95. if (opers.length > 1) {
  96. // Mol* operators are right-to-left
  97. return opers[1] + 'x' + opers[0];
  98. }
  99. return opers[0];
  100. }
  101. export async function uploadStructure(ctx: StrucmotifCtx) {
  102. const { entryId, plugin, structure } = ctx;
  103. const formData = new FormData();
  104. formData.append('format', 'bcif');
  105. const name = entryId.replace(/\W/g, '') || 'unknown';
  106. formData.append('name', name);
  107. const file = new File([to_mmCIF(name, structure, true, { copyAllCategories: true })], name + '.bcif');
  108. formData.append('file', file);
  109. try {
  110. const res = await fetch(FILE_STORAGE_PUT_URL, { method: 'POST', body: formData });
  111. if (!res.ok || res.status !== 200) {
  112. plugin.log.warn('File Upload Failed!');
  113. return void 0;
  114. }
  115. const { key } = await res.json();
  116. const url = FILE_STORAGE_GET_URL + key;
  117. plugin.log.info(`Uploaded File is at: ${url}`);
  118. return url;
  119. } catch (e) {
  120. plugin.log.warn('File Upload Failed!');
  121. return void 0;
  122. }
  123. }