123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157 |
- /**
- * Copyright (c) 2018 mol* contributors, licensed under MIT, See LICENSE file for more info.
- *
- * @author David Sehnal <david.sehnal@gmail.com>
- */
- import { Column, Table } from 'mol-data/db';
- import { toTable } from 'mol-io/reader/cif/schema';
- import { mmCIF_residueId_schema } from 'mol-io/reader/cif/schema/mmcif-extras';
- import { CifWriter } from 'mol-io/writer/cif';
- import { Model, ModelPropertyDescriptor, ResidueIndex, StructureProperties as P, Unit, IndexedCustomProperty } from 'mol-model/structure';
- import { residueIdFields } from 'mol-model/structure/export/categories/atom_site';
- import { StructureElement } from 'mol-model/structure/structure';
- import { CustomPropSymbol } from 'mol-script/language/symbol';
- import Type from 'mol-script/language/type';
- import { QuerySymbolRuntime } from 'mol-script/runtime/query/compiler';
- import { PropertyWrapper } from '../common/wrapper';
- import CifField = CifWriter.Field;
- export namespace StructureQualityReport {
- export type IssueMap = IndexedCustomProperty.Residue<string[]>
- export type Property = PropertyWrapper<IssueMap | undefined>
- export function get(model: Model): Property | undefined {
- // must be defined before the descriptor so it's not undefined.
- return model._dynamicPropertyData.__StructureQualityReport__;
- }
- export const Descriptor = ModelPropertyDescriptor({
- isStatic: false,
- name: 'structure_quality_report',
- cifExport: {
- prefix: 'pdbe',
- categories: [
- PropertyWrapper.defaultInfoCategory('pdbe_structure_quality_report', StructureQualityReport.get),
- {
- name: 'pdbe_structure_quality_report_issues',
- instance(ctx) {
- return {
- fields: _structure_quality_report_issues_fields,
- source: ctx.structures.map(s => IndexedCustomProperty.getCifDataSource(s, StructureQualityReport.getIssueMap(s.model), ctx.cache))
- };
- }
- }]
- },
- symbols: {
- issueCount: QuerySymbolRuntime.Dynamic(CustomPropSymbol('pdbe', 'structure-quality.issue-count', Type.Num),
- ctx => StructureQualityReport.getIssues(ctx.element).length),
- // TODO: add (hasIssue :: IssueType(extends string) -> boolean) symbol
- }
- });
- export const Schema = {
- pdbe_structure_quality_report: {
- updated_datetime_utc: Column.Schema.str
- },
- pdbe_structure_quality_report_issues: {
- id: Column.Schema.int,
- ...mmCIF_residueId_schema,
- pdbx_PDB_model_num: Column.Schema.int,
- issues: Column.Schema.List(',', x => x)
- }
- }
- function getCifData(model: Model) {
- if (model.sourceData.kind !== 'mmCIF') throw new Error('Data format must be mmCIF.');
- return toTable(Schema.pdbe_structure_quality_report_issues, model.sourceData.frame.categories.pdbe_structure_quality_report_issues);
- }
- export async function attachFromCifOrApi(model: Model, params: {
- // provide JSON from api
- PDBe_apiSourceJson?: (model: Model) => Promise<any>
- }) {
- if (get(model)) return true;
- let issueMap: IssueMap | undefined;
- let info = PropertyWrapper.tryGetInfoFromCif('pdbe_structure_quality_report', model);
- if (info) {
- const data = getCifData(model);
- issueMap = createIssueMapFromCif(model, data);
- } else if (params.PDBe_apiSourceJson) {
- const data = await params.PDBe_apiSourceJson(model);
- if (!data) return false;
- info = PropertyWrapper.createInfo();
- issueMap = createIssueMapFromJson(model, data);
- } else {
- return false;
- }
- model.customProperties.add(Descriptor);
- set(model, { info, data: issueMap });
- return true;
- }
- function set(model: Model, prop: Property) {
- (model._dynamicPropertyData.__StructureQualityReport__ as Property) = prop;
- }
- export function getIssueMap(model: Model): IssueMap | undefined {
- const prop = get(model);
- return prop && prop.data;
- }
- const _emptyArray: string[] = [];
- export function getIssues(e: StructureElement) {
- if (!Unit.isAtomic(e.unit)) return _emptyArray;
- const prop = StructureQualityReport.get(e.unit.model);
- if (!prop || !prop.data) return _emptyArray;
- const rI = e.unit.residueIndex[e.element];
- return prop.data.has(rI) ? prop.data.get(rI)! : _emptyArray;
- }
- }
- type ExportCtx = IndexedCustomProperty.ExportCtx<string[]>
- const _structure_quality_report_issues_fields: CifField<number, ExportCtx>[] = CifWriter.fields()
- .index('id')
- .many(residueIdFields((i, d) => d.elements[i]))
- .int('pdbx_PDB_model_num', (i, d) => P.unit.model_num(d.elements[i]))
- .str('issues', (i, d) => d.property(i).join(','))
- .getFields()
- function createIssueMapFromJson(modelData: Model, data: any): StructureQualityReport.IssueMap | undefined {
- const ret = new Map<ResidueIndex, string[]>();
- if (!data.molecules) return;
- for (const entity of data.molecules) {
- const entity_id = entity.entity_id.toString();
- for (const chain of entity.chains) {
- const asym_id = chain.struct_asym_id.toString();
- for (const model of chain.models) {
- const model_id = model.model_id.toString();
- if (+model_id !== modelData.modelNum) continue;
- for (const residue of model.residues) {
- const auth_seq_id = residue.author_residue_number, ins_code = residue.author_insertion_code || '';
- const idx = modelData.atomicHierarchy.index.findResidue(entity_id, asym_id, auth_seq_id, ins_code);
- ret.set(idx, residue.outlier_types);
- }
- }
- }
- }
- return IndexedCustomProperty.fromResidueMap(ret);
- }
- function createIssueMapFromCif(modelData: Model, data: Table<typeof StructureQualityReport.Schema.pdbe_structure_quality_report_issues>): StructureQualityReport.IssueMap | undefined {
- const ret = new Map<ResidueIndex, string[]>();
- const { label_entity_id, label_asym_id, auth_seq_id, pdbx_PDB_ins_code, issues, pdbx_PDB_model_num, _rowCount } = data;
- for (let i = 0; i < _rowCount; i++) {
- if (pdbx_PDB_model_num.value(i) !== modelData.modelNum) continue;
- const idx = modelData.atomicHierarchy.index.findResidue(label_entity_id.value(i), label_asym_id.value(i), auth_seq_id.value(i), pdbx_PDB_ins_code.value(i));
- ret.set(idx, issues.value(i));
- }
- return IndexedCustomProperty.fromResidueMap(ret);
- }
|