123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252 |
- /**
- * 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) {
- const prop = get(ctx.firstModel);
- if (!prop) return CifWriter.Category.Empty;
- let groupCtx: ReportExportContext;
- if (ctx.cache.pdbe_structure_quality_report_issues) groupCtx = ctx.cache.pdbe_structure_quality_report_issues;
- else {
- const exportCtx = prop.data!.getExportContext(ctx.structures[0]);
- groupCtx = createExportContext(exportCtx);
- ctx.cache.pdbe_structure_quality_report_issues = groupCtx;
- }
- return {
- fields: _structure_quality_report_issues_fields,
- source: [{ data: groupCtx, rowCount: groupCtx.elements.length }]
- }
- // return {
- // fields: _structure_quality_report_issues_fields,
- // source: ctx.structures.map(s => IndexedCustomProperty.getCifDataSource(s, StructureQualityReport.getIssueMap(s.model), ctx.cache))
- // };
- }
- }, {
- name: 'pdbe_structure_quality_report_issue_types',
- instance(ctx) {
- const prop = get(ctx.firstModel);
- if (!prop) return CifWriter.Category.Empty;
- let groupCtx: ReportExportContext;
- if (ctx.cache.pdbe_structure_quality_report_issues) groupCtx = ctx.cache.pdbe_structure_quality_report_issues;
- else {
- const exportCtx = prop.data!.getExportContext(ctx.structures[0]);
- groupCtx = createExportContext(exportCtx);
- ctx.cache.pdbe_structure_quality_report_issues = groupCtx;
- }
- return {
- fields: _structure_quality_report_issue_types_fields,
- source: [{ data: groupCtx, rowCount: groupCtx.rows.length }]
- }
- // 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,
- issue_group_id: Column.Schema.int
- },
- pdbe_structure_quality_report_issue_types: {
- group_id: Column.Schema.int,
- issue_type: Column.Schema.str
- }
- }
- function getCifData(model: Model) {
- if (model.sourceData.kind !== 'mmCIF') throw new Error('Data format must be mmCIF.');
- return {
- residues: toTable(Schema.pdbe_structure_quality_report_issues, model.sourceData.frame.categories.pdbe_structure_quality_report_issues),
- groups: toTable(Schema.pdbe_structure_quality_report_issue_types, model.sourceData.frame.categories.pdbe_structure_quality_report_issue_types),
- }
- }
- 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.residues, data.groups);
- } 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, ReportExportContext>[] = CifWriter.fields<number, ReportExportContext>()
- .index('id')
- .many(residueIdFields((i, d) => d.elements[i], { includeModelNum: true }))
- .int('group_id', (i, d) => d.groupId[i])
- .getFields();
- interface ReportExportContext extends ExportCtx {
- groupId: number[],
- rows: [number, string][]
- }
- const _structure_quality_report_issue_types_fields: CifField<number, ReportExportContext>[] = CifWriter.fields<number, ReportExportContext>()
- .int('group_id', (i, d) => d.rows[i][0])
- .str('issue_type', (i, d) => d.rows[i][1])
- .getFields();
- function createExportContext(ctx: ExportCtx): ReportExportContext {
- const groupMap = new Map<string, number>();
- const groupId: number[] = [];
- const rows: ReportExportContext['rows'] = [];
- for (let i = 0; i < ctx.elements.length; i++) {
- const issues = ctx.property(i);
- const key = issues.join(',');
- if (!groupMap.has(key)) {
- const idx = groupMap.size + 1;
- groupMap.set(key, idx);
- for (const issue of issues) {
- rows.push([idx, issue]);
- }
- }
- groupId[i] = groupMap.get(key)!;
- }
- return { ...ctx, groupId, rows };
- }
- 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,
- residueData: Table<typeof StructureQualityReport.Schema.pdbe_structure_quality_report_issues>,
- groupData: Table<typeof StructureQualityReport.Schema.pdbe_structure_quality_report_issue_types>): StructureQualityReport.IssueMap | undefined {
- const ret = new Map<ResidueIndex, string[]>();
- const { label_entity_id, label_asym_id, auth_seq_id, pdbx_PDB_ins_code, issue_group_id, pdbx_PDB_model_num, _rowCount } = residueData;
- const groups = parseIssueTypes(groupData);
- 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, groups.get(issue_group_id.value(i))!);
- }
- return IndexedCustomProperty.fromResidueMap(ret);
- }
- function parseIssueTypes(groupData: Table<typeof StructureQualityReport.Schema.pdbe_structure_quality_report_issue_types>): Map<number, string[]> {
- const ret = new Map<number, string[]>();
- const { group_id, issue_type } = groupData;
- for (let i = 0; i < groupData._rowCount; i++) {
- let group: string[];
- const id = group_id.value(i);
- if (ret.has(id)) group = ret.get(id)!;
- else {
- group = [];
- ret.set(id, group);
- }
- group.push(issue_type.value(i));
- }
- return ret;
- }
|