Skip to content

Commit

Permalink
Browse files Browse the repository at this point in the history
…raph_handler into pathfinder
  • Loading branch information
tokebe committed Apr 17, 2024
2 parents 82bd2d6 + dc48864 commit 78c0c84
Show file tree
Hide file tree
Showing 15 changed files with 80 additions and 194 deletions.
10 changes: 5 additions & 5 deletions src/cache_handler.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ import chunker from 'stream-chunker';
import { Readable, Transform } from 'stream';
import { Record, RecordPackage } from '@biothings-explorer/api-response-transform';
import { threadId } from 'worker_threads';
import MetaKG from '../../smartapi-kg/built';
import MetaKG from '@biothings-explorer/smartapi-kg';
import QEdge from './query_edge';
import { QueryHandlerOptions } from '@biothings-explorer/types';

Expand Down Expand Up @@ -113,7 +113,7 @@ export default class CacheHandler {
}

async categorizeEdges(qEdges: QEdge[]): Promise<{ cachedRecords: Record[]; nonCachedQEdges: QEdge[] }> {
if (this.cacheEnabled === false || process.env.INTERNAL_DISABLE_REDIS === "true") {
if (this.cacheEnabled === false || process.env.INTERNAL_DISABLE_REDIS === 'true') {
return {
cachedRecords: [],
nonCachedQEdges: qEdges,
Expand Down Expand Up @@ -210,7 +210,7 @@ export default class CacheHandler {
}

async cacheEdges(queryRecords: Record[]): Promise<void> {
if (this.cacheEnabled === false || process.env.INTERNAL_DISABLE_REDIS === "true") {
if (this.cacheEnabled === false || process.env.INTERNAL_DISABLE_REDIS === 'true') {
if (global.parentPort) {
global.parentPort.postMessage({ threadId, cacheDone: true });
}
Expand Down Expand Up @@ -257,8 +257,8 @@ export default class CacheHandler {
resolve();
});
});
if (process.env.QEDGE_CACHE_TIME_S !== "0") {
await redisClient.client.expireTimeout(redisID, process.env.QEDGE_CACHE_TIME_S || 1800);
if (process.env.QEDGE_CACHE_TIME_S !== '0') {
await redisClient.client.expireTimeout(redisID, process.env.QEDGE_CACHE_TIME_S || 1800);
}
} catch (error) {
failedHashes.push(hash);
Expand Down
10 changes: 8 additions & 2 deletions src/graph/graph.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,8 @@ import Debug from 'debug';
import { LogEntry, StampedLog } from '@biothings-explorer/utils';
import KGNode from './kg_node';
import KGEdge from './kg_edge';
import { Record } from '../../../api-response-transform/built';
import { TrapiAuxiliaryGraph, TrapiResult } from '../types';
import { Record } from '@biothings-explorer/api-response-transform';
import { TrapiAuxiliaryGraph, TrapiResult } from '@biothings-explorer/types';
import KnowledgeGraph from './knowledge_graph';
const debug = Debug('bte:biothings-explorer-trapi:Graph');

Expand Down Expand Up @@ -104,6 +104,12 @@ export default class BTEGraph {
.map((item) => {
this.edges[recordHash].addAdditionalAttributes(item, record.mappedResponse[item]);
});
if (record.knowledge_level) {
this.edges[recordHash].addAdditionalAttributes('biolink:knowledge_level', record.knowledge_level);
}
if (record.agent_type) {
this.edges[recordHash].addAdditionalAttributes('biolink:agent_type', record.agent_type);
}
this.edges[recordHash].addSource(record.provenanceChain);
Object.entries(record.qualifiers).forEach(([qualifierType, qualifier]) => {
this.edges[recordHash].addQualifier(qualifierType, qualifier);
Expand Down
20 changes: 14 additions & 6 deletions src/graph/kg_edge.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { ProvenanceChainItem } from '@biothings-explorer/api-response-transform';
import { TrapiAttribute } from '../types';
import { TrapiSource } from '@biothings-explorer/types';
import { TrapiAttribute } from '@biothings-explorer/types';

export interface KGEdgeInfo {
object: string;
Expand All @@ -20,6 +20,7 @@ export default class KGEdge {
resource_id: string;
resource_role: string;
upstream_resource_ids?: Set<string>;
source_record_urls?: Set<string>;
};
};
};
Expand Down Expand Up @@ -68,7 +69,7 @@ export default class KGEdge {
});
}

addSource(source: ProvenanceChainItem | ProvenanceChainItem[]): void {
addSource(source: TrapiSource | TrapiSource[]): void {
if (typeof source === 'undefined') {
return;
}
Expand All @@ -77,19 +78,26 @@ export default class KGEdge {
}
source.forEach((item) => {
if (!this.sources[item.resource_id]) this.sources[item.resource_id] = {};
if (item.upstream_resource_ids && !Array.isArray(item.upstream_resource_ids)) {
item.upstream_resource_ids = [item.upstream_resource_ids];
}
if (item.source_record_urls && !Array.isArray(item.source_record_urls)) {
item.source_record_urls = [item.source_record_urls];
}
if (!this.sources[item.resource_id][item.resource_role]) {
this.sources[item.resource_id][item.resource_role] = {
resource_id: item.resource_id,
resource_role: item.resource_role,
upstream_resource_ids: item.upstream_resource_ids ? new Set(item.upstream_resource_ids) : undefined,
source_record_urls: item.source_record_urls ? new Set(item.source_record_urls) : undefined,
};
}
if (item.upstream_resource_ids && !Array.isArray(item.upstream_resource_ids)) {
item.upstream_resource_ids = [item.upstream_resource_ids];
}
item.upstream_resource_ids?.forEach((upstream) =>
this.sources[item.resource_id][item.resource_role].upstream_resource_ids.add(upstream),
);
item.source_record_urls?.forEach((url) =>
this.sources[item.resource_id][item.resource_role].source_record_urls.add(url),
);
});
}

Expand Down
2 changes: 1 addition & 1 deletion src/graph/kg_node.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { TrapiAttribute } from '../types';
import { TrapiAttribute } from '@biothings-explorer/types';

export interface KGNodeInfo {
label: string;
Expand Down
12 changes: 9 additions & 3 deletions src/graph/knowledge_graph.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,14 +9,16 @@ import {
TrapiKGNodes,
TrapiQualifier,
TrapiSource,
} from '../types';
} from '@biothings-explorer/types';
import KGNode from './kg_node';
import KGEdge from './kg_edge';
import { BTEGraphUpdate } from './graph';
import { APIDefinition } from '@biothings-explorer/types';

const debug = Debug('bte:biothings-explorer-trapi:KnowledgeGraph');

const NON_ARRAY_ATTRIBUTES = ['biolink:knowledge_level', 'biolink:agent_type'];

export default class KnowledgeGraph {
nodes: {
[nodePrimaryID: string]: TrapiKGNode;
Expand Down Expand Up @@ -114,11 +116,14 @@ export default class KnowledgeGraph {
}

Object.entries(kgEdge.attributes).forEach(([key, value]) => {
if (key == 'edge-attributes') return;
if (key === 'edge-attributes') return;
// if (key == 'edge-attributes') return;
attributes.push({
attribute_type_id: key,
value: Array.from(value as Set<string>),
value: // technically works for numbers as well
NON_ARRAY_ATTRIBUTES.includes(key)
? [...(value as Set<string>)].reduce((acc, val) => acc + val)
: Array.from(value as Set<string>),
//value_type_id: 'bts:' + key,
});
});
Expand All @@ -137,6 +142,7 @@ export default class KnowledgeGraph {
const trapiSource: TrapiSource = {
...sourceObj,
upstream_resource_ids: sourceObj.upstream_resource_ids ? [...sourceObj.upstream_resource_ids] : undefined,
source_record_urls: sourceObj.source_record_urls ? [...sourceObj.source_record_urls] : undefined,
};
sources.push(trapiSource);
});
Expand Down
14 changes: 9 additions & 5 deletions src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ import {
TrapiQueryGraph,
TrapiResponse,
TrapiResult,
} from './types';
} from '@biothings-explorer/types';
import { QueryHandlerOptions } from '@biothings-explorer/types';
import BTEGraph from './graph/graph';
import QEdge from './query_edge';
Expand Down Expand Up @@ -174,6 +174,8 @@ export default class TRAPIQueryHandler {
const source = Object.entries(ontologyKnowledgeSourceMapping).find(([prefix]) => {
return expanded.includes(prefix);
})[1];
subclassEdge.addAdditionalAttributes('biolink:knowledge_level', 'knowledge_assertion')
subclassEdge.addAdditionalAttributes('biolink:agent_type', 'manual_agent')
subclassEdge.addSource([
{ resource_id: source, resource_role: 'primary_knowledge_source' },
{
Expand Down Expand Up @@ -210,7 +212,7 @@ export default class TRAPIQueryHandler {
suffix += 1;
}
const supportGraphID = `support${suffix}-${boundEdgeID}`;
auxGraphs[supportGraphID] = { edges: supportGraph };
auxGraphs[supportGraphID] = { edges: supportGraph, attributes: [] };
if (!edgesIDsByAuxGraphID[supportGraphID]) {
edgesIDsByAuxGraphID[supportGraphID] = new Set();
}
Expand All @@ -222,6 +224,8 @@ export default class TRAPIQueryHandler {
object: object,
});
boundEdge.addAdditionalAttributes('biolink:support_graphs', [supportGraphID]);
boundEdge.addAdditionalAttributes('biolink:knowledge_level', 'logical_entailment')
boundEdge.addAdditionalAttributes('biolink:agent_type', 'automated_agent')
boundEdge.addSource([
{
resource_id: this.options.provenanceUsesServiceProvider
Expand Down Expand Up @@ -251,7 +255,7 @@ export default class TRAPIQueryHandler {
boundIDs.add(binding.id);
}
} else if (!boundIDs.has(nodesToRebind[binding.id].newNode)) {
newBindings.push({ id: nodesToRebind[binding.id].newNode });
newBindings.push({ id: nodesToRebind[binding.id].newNode, attributes: [] });
boundIDs.add(nodesToRebind[binding.id].newNode);
}
return { boundIDs, newBindings };
Expand All @@ -273,7 +277,7 @@ export default class TRAPIQueryHandler {
boundIDs.add(binding.id);
}
} else if (!boundIDs.has(edgesToRebind[binding.id])) {
newBindings.push({ id: edgesToRebind[binding.id] });
newBindings.push({ id: edgesToRebind[binding.id], attributes: [] });
boundIDs.add(edgesToRebind[binding.id]);
resultBoundEdgesWithAuxGraphs.add(edgesToRebind[binding.id]);
}
Expand Down Expand Up @@ -334,7 +338,7 @@ export default class TRAPIQueryHandler {
description: `Query processed successfully, retrieved ${results.length} results.`,
schema_version: global.SCHEMA_VERSION,
biolink_version: global.BIOLINK_VERSION,
workflow: [{ id: 'lookup' }],
workflow: [{ id: this.options.smartAPIID || this.options.teamName ? 'lookup' : 'lookup_and_score' }],
message: {
query_graph: this.originalQueryGraph,
knowledge_graph: this.knowledgeGraph.kg,
Expand Down
37 changes: 22 additions & 15 deletions src/inferred_mode/inferred_mode.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,9 +6,8 @@ import biolink from '../biolink';
import { getTemplates, MatchedTemplate, TemplateLookup } from './template_lookup';
import { scaled_sigmoid, inverse_scaled_sigmoid } from '../results_assembly/score';
import TRAPIQueryHandler from '../index';
import { QueryHandlerOptions } from '@biothings-explorer/types';
import {
CompactQualifiers,
QueryHandlerOptions,
TrapiAuxGraphCollection,
TrapiEdgeBinding,
TrapiKnowledgeGraph,
Expand All @@ -18,7 +17,8 @@ import {
TrapiResponse,
TrapiResult,
TrapiAnalysis
} from '../types';
} from '@biothings-explorer/types';
import { CompactQualifiers } from '../index';
const debug = Debug('bte:biothings-explorer-trapi:inferred-mode');

export interface CombinedResponse {
Expand Down Expand Up @@ -298,8 +298,8 @@ export default class InferredQueryHandler {
newResponse.message.results.forEach((result) => {
const translatedResult: TrapiResult = {
node_bindings: {
[qEdge.subject]: [{ id: result.node_bindings.creativeQuerySubject[0].id }],
[qEdge.object]: [{ id: result.node_bindings.creativeQueryObject[0].id }],
[qEdge.subject]: [{ id: result.node_bindings.creativeQuerySubject[0].id, attributes: [] }],
[qEdge.object]: [{ id: result.node_bindings.creativeQueryObject[0].id, attributes: [] }],
},
pfocr: result.pfocr?.length ? result.pfocr : undefined,
analyses: [
Expand Down Expand Up @@ -333,14 +333,14 @@ export default class InferredQueryHandler {
// Direct edge answers stand on their own, not as an inferred edge.
if (Object.keys(result.node_bindings).length == 2) {
const boundEdgeID = Object.values(result.analyses[0].edge_bindings)[0][0].id;
translatedResult.analyses[0].edge_bindings = { [qEdgeID]: [{ id: boundEdgeID }] };
translatedResult.analyses[0].edge_bindings = { [qEdgeID]: [{ id: boundEdgeID, attributes: [] }] };
} else {
// Create an aux graph using the result and associate it with an inferred Edge
const inferredEdgeID = `inferred-${resultCreativeSubjectID}-${qEdge.predicates[0].replace(
'biolink:',
'',
)}-${resultCreativeObjectID}`;
translatedResult.analyses[0].edge_bindings = { [qEdgeID]: [{ id: inferredEdgeID }] };
translatedResult.analyses[0].edge_bindings = { [qEdgeID]: [{ id: inferredEdgeID, attributes: [] }] };
if (!combinedResponse.message.knowledge_graph.edges[inferredEdgeID]) {
combinedResponse.message.knowledge_graph.edges[inferredEdgeID] = {
subject: resultCreativeSubjectID,
Expand All @@ -354,7 +354,11 @@ export default class InferredQueryHandler {
resource_role: 'primary_knowledge_source',
},
],
attributes: [{ attribute_type_id: 'biolink:support_graphs', value: [] }],
attributes: [
{ attribute_type_id: 'biolink:support_graphs', value: [] },
{ attribute_type_id: 'biolink:knowledge_level', value: "prediction" },
{ attribute_type_id: 'biolink:agent_type', value: "computational_model" },
],
};
}
if (!auxGraphSuffixes[inferredEdgeID]) auxGraphSuffixes[inferredEdgeID] = 0;
Expand All @@ -371,6 +375,7 @@ export default class InferredQueryHandler {
},
[] as string[],
),
attributes: []
};

if (this.pathfinder) {
Expand Down Expand Up @@ -407,9 +412,9 @@ export default class InferredQueryHandler {
if (typeof combinedResponse.message.results[resultID].analyses[0].score !== 'undefined') {
combinedResponse.message.results[resultID].analyses[0].score = resScore
? scaled_sigmoid(
inverse_scaled_sigmoid(combinedResponse.message.results[resultID].analyses[0].score) +
inverse_scaled_sigmoid(resScore),
)
inverse_scaled_sigmoid(combinedResponse.message.results[resultID].analyses[0].score) +
inverse_scaled_sigmoid(resScore),
)
: combinedResponse.message.results[resultID].analyses[0].score;
} else {
combinedResponse.message.results[resultID].analyses[0].score = resScore;
Expand Down Expand Up @@ -521,7 +526,7 @@ export default class InferredQueryHandler {
description: '',
schema_version: global.SCHEMA_VERSION,
biolink_version: global.BIOLINK_VERSION,
workflow: [{ id: 'lookup' }],
workflow: [{ id: 'lookup_and_score' }],
message: {
query_graph: this.queryGraph,
knowledge_graph: {
Expand Down Expand Up @@ -582,7 +587,8 @@ export default class InferredQueryHandler {
`creative result maximum of ${this.CREATIVE_LIMIT} (reaching ${
creativeLimitHit
} merged). `,
`Response will be truncated to top-scoring ${this.CREATIVE_LIMIT} results. Skipping remaining ${subQueries.length - (i + 1)
`Response will be truncated to top-scoring ${this.CREATIVE_LIMIT} results. Skipping remaining ${
subQueries.length - (i + 1)
} `,
subQueries.length - (i + 1) === 1 ? `template.` : `templates.`,
].join('');
Expand All @@ -607,8 +613,9 @@ export default class InferredQueryHandler {
const total =
Object.values(mergedResultsCount).reduce((sum, count) => sum + count, 0) +
Object.keys(mergedResultsCount).length;
const message = `Merging Summary: (${total}) inferred-template results were merged into (${Object.keys(mergedResultsCount).length
}) final results, reducing result count by (${total - Object.keys(mergedResultsCount).length})`;
const message = `Merging Summary: (${total}) inferred-template results were merged into (${
Object.keys(mergedResultsCount).length
}) final results, reducing result count by (${total - Object.keys(mergedResultsCount).length})`;
debug(message);
combinedResponse.logs.push(new LogEntry('INFO', null, message).getLog());
}
Expand Down
5 changes: 2 additions & 3 deletions src/inferred_mode/pathfinder.ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,8 @@
import TRAPIQueryHandler from '../index';
import { TrapiResponse, TrapiQEdge, TrapiResult, TrapiQueryGraph, TrapiQNode, TrapiAnalysis } from '../types';
import { TrapiResponse, TrapiQEdge, TrapiResult, TrapiQueryGraph, TrapiQNode, TrapiAnalysis, QueryHandlerOptions } from '@biothings-explorer/types';
import InferredQueryHandler from './inferred_mode';
import { scaled_sigmoid, inverse_scaled_sigmoid } from '../results_assembly/score';
import { LogEntry, StampedLog, Telemetry } from '@biothings-explorer/utils';
import { QueryHandlerOptions } from '@biothings-explorer/types';
import Debug from 'debug';
const debug = Debug('bte:biothings-explorer-trapi:pathfinder');

Expand Down Expand Up @@ -313,4 +312,4 @@ export default class PathfinderQueryHandler {

return { results: newResultObject, graphs: newAuxGraphs };
}
}
}
3 changes: 2 additions & 1 deletion src/inferred_mode/template_lookup.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,8 @@
import { promises as fs } from 'fs';
import path from 'path';
import async from 'async';
import { TrapiQueryGraph, CompactQualifiers } from '../types';
import { TrapiQueryGraph } from '@biothings-explorer/types';
import { CompactQualifiers } from '../types';

export interface TemplateLookup {
subject: string;
Expand Down
2 changes: 1 addition & 1 deletion src/query_edge.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ import { Record, RecordNode, FrozenRecord } from '@biothings-explorer/api-respon
import QNode from './query_node';
import { QNodeInfo } from './query_node';
import { LogEntry, StampedLog } from '@biothings-explorer/utils';
import { TrapiAttributeConstraint, TrapiQualifierConstraint } from './types';
import { TrapiAttributeConstraint, TrapiQualifierConstraint } from '@biothings-explorer/types';

const debug = Debug('bte:biothings-explorer-trapi:QEdge');

Expand Down
2 changes: 1 addition & 1 deletion src/query_graph.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ import biolink from './biolink';
import { resolveSRI } from 'biomedical_id_resolver';
import _ from 'lodash';
import * as utils from './utils';
import { TrapiQueryGraph } from './types';
import { TrapiQueryGraph } from '@biothings-explorer/types';

const debug = Debug('bte:biothings-explorer-trapi:query_graph');

Expand Down
2 changes: 1 addition & 1 deletion src/results_assembly/pfocr.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ const debug = Debug('bte:biothings-explorer-trapi:pfocr');
import { intersection } from '../utils';
import _ from 'lodash';
import { LogEntry, StampedLog } from '@biothings-explorer/utils';
import { TrapiResult } from '../types';
import { TrapiResult } from '@biothings-explorer/types';

// the minimum acceptable intersection size between the CURIEs
// in a TRAPI result and in a PFOCR figure.
Expand Down
Loading

0 comments on commit 78c0c84

Please sign in to comment.