Skip to content

Commit

Permalink
feat: Cache parsed SQL when building sequence diagram
Browse files Browse the repository at this point in the history
  • Loading branch information
ahtrotta committed Oct 9, 2023
1 parent 13ca5f1 commit a6b3766
Show file tree
Hide file tree
Showing 5 changed files with 45 additions and 22 deletions.
1 change: 1 addition & 0 deletions packages/models/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@
"eslint-plugin-import": "^2.22.1",
"jest": "^27.4.7",
"lint-staged": "^10.5.4",
"lru-cache": "6.0.0",
"prettier": "^2.7.1",
"ts-jest": "^27.1.4",
"tsup": "^6.5.0",
Expand Down
35 changes: 26 additions & 9 deletions packages/models/src/event.js
Original file line number Diff line number Diff line change
Expand Up @@ -362,14 +362,14 @@ export default class Event {

get identityHash() {
if (!this.$hidden.identityHash) {
this.$hidden.identityHash = this.buildIdentityHash(this).digest();
this.$hidden.identityHash = this.buildIdentityHash().digest();
}
return this.$hidden.identityHash;
}

get hash() {
if (!this.$hidden.hash) {
this.$hidden.hash = this.buildStableHash(this).digest();
this.$hidden.hash = this.buildStableHash().digest();
}
return this.$hidden.hash;
}
Expand All @@ -381,6 +381,13 @@ export default class Event {
return this.$hidden.stableProperties;
}

getHashWithSqlCache(parsedSqlCache) {

This comment has been minimized.

Copy link
@kgilpin

kgilpin Oct 9, 2023

Contributor

I think this can be removed, and clients that want to provide a cache can just call buildStableHash(cache).digest();. Because it's kind of a combination builder/getter and it's not clear from the interface when parsedSqlCache is used and whether or not it's rquired.

if (!this.$hidden.hash) {
this.$hidden.hash = this.buildStableHash(parsedSqlCache).digest();
}
return this.$hidden.hash;
}

callStack() {
const stack = this.ancestors().reverse();
stack.push(this.callEvent);
Expand Down Expand Up @@ -501,7 +508,7 @@ export default class Event {

// Collects properties of an event which are not dependent on the specifics
// of invocation.
gatherStableProperties() {
gatherStableProperties(parsedSqlCache) {

This comment has been minimized.

Copy link
@kgilpin

kgilpin Oct 9, 2023

Contributor

Is the cache allowed to be undefined? Unclear since it's JS.

const { sqlQuery } = this;

// Convert null and undefined values to empty strings
Expand All @@ -526,10 +533,17 @@ export default class Event {

let properties;
if (sqlQuery) {
const sqlNormalized = abstractSqlAstJSON(sqlQuery, this.sql.database_type)
// Collapse repeated variable literals and parameter tokens (e.g. '?, ?' in an IN clause)
.split(/{"type":"variable"}(?:,{"type":"variable"})*/g)
.join(`{"type":"variable"}`);
let sqlNormalized;
const cacheKey = `${this.sql.database_type}:${sqlQuery}`;
if (parsedSqlCache) sqlNormalized = parsedSqlCache.get(cacheKey);
if (!sqlNormalized) {
sqlNormalized = abstractSqlAstJSON(sqlQuery, this.sql.database_type)
// Collapse repeated variable literals and parameter tokens (e.g. '?, ?' in an IN clause)
.split(/{"type":"variable"}(?:,{"type":"variable"})*/g)
.join(`{"type":"variable"}`);
parsedSqlCache.set(cacheKey, sqlNormalized);
}

properties = {
event_type: 'sql',
sql_normalized: sqlNormalized,
Expand All @@ -554,7 +568,10 @@ export default class Event {
return HashBuilder.buildHash('event-identity-v2', this.gatherIdentityProperties());
}

buildStableHash() {
return HashBuilder.buildHash('event-stable-properties-v2', this.gatherStableProperties());
buildStableHash(parsedSqlCache) {
return HashBuilder.buildHash(
'event-stable-properties-v2',
this.gatherStableProperties(parsedSqlCache)
);
}
}
2 changes: 2 additions & 0 deletions packages/models/types/index.d.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import type { SqliteParser } from './sqlite-parser';
import type LRUCache from 'lru-cache';

declare module '@appland/models' {
export type CodeObjectType =
Expand Down Expand Up @@ -214,6 +215,7 @@ declare module '@appland/models' {
dataObjects(): Array<ParameterObject | ReturnValueObject>;
toString(): string;
toJSON(): any;
getHashWithSqlCache(parsedSqlCache: LRUCache<string, any>): string;
}

export class EventNavigator {
Expand Down
10 changes: 6 additions & 4 deletions packages/sequence-diagram/src/buildDiagram.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import { AppMap, Event } from '@appland/models';
import { classNameToOpenAPIType } from '@appland/openapi';
import sha256 from 'crypto-js/sha256.js';
import LRUCache from 'lru-cache';
import { merge } from './mergeWindow';
import { selectEvents } from './selectEvents';
import Specification from './specification';
Expand All @@ -19,6 +20,7 @@ import {
} from './types';

const MAX_WINDOW_SIZE = 5;
const parsedSqlCache = new LRUCache<string, any>({ max: 1000 });

class ActorManager {
private _actorsByCodeObjectId = new Map<string, Actor>();
Expand Down Expand Up @@ -68,7 +70,7 @@ export default function buildDiagram(
callee: actorManager.findOrCreateActor(callee),
route: callee.route,
status: response.status || response.status_code,
digest: callee.hash,
digest: callee.getHashWithSqlCache(parsedSqlCache),
subtreeDigest: 'undefined',
children: [],
elapsed: callee.elapsedTime,
Expand All @@ -83,7 +85,7 @@ export default function buildDiagram(
callee: actorManager.findOrCreateActor(callee),
route: callee.route,
status: response.status || response.status_code,
digest: callee.hash,
digest: callee.getHashWithSqlCache(parsedSqlCache),
subtreeDigest: 'undefined',
children: [],
elapsed: callee.elapsedTime,
Expand All @@ -96,7 +98,7 @@ export default function buildDiagram(
caller: caller ? actorManager.findOrCreateActor(caller) : undefined,
callee: actorManager.findOrCreateActor(callee),
query: callee.sqlQuery,
digest: truncatedQuery ? 'truncatedQuery' : callee.hash,
digest: truncatedQuery ? 'truncatedQuery' : callee.getHashWithSqlCache(parsedSqlCache),

This comment has been minimized.

Copy link
@kgilpin

kgilpin Oct 9, 2023

Contributor

If it's a truncated query, maybe we should be using the sha256 of the query string. Otherwise all truncated queries will be treated as identical for digest purposes. I don't know, it's a tough case to handle.

subtreeDigest: 'undefined',
children: [],
elapsed: callee.elapsedTime,
Expand All @@ -109,7 +111,7 @@ export default function buildDiagram(
callee: actorManager.findOrCreateActor(callee),
name: callee.codeObject.name,
static: callee.codeObject.static,
digest: callee.hash,
digest: callee.getHashWithSqlCache(parsedSqlCache),
subtreeDigest: 'undefined',
stableProperties: { ...callee.stableProperties },
returnValue: buildReturnValue(callee),
Expand Down
19 changes: 10 additions & 9 deletions yarn.lock
Original file line number Diff line number Diff line change
Expand Up @@ -380,6 +380,7 @@ __metadata:
eslint-plugin-import: ^2.22.1
jest: ^27.4.7
lint-staged: ^10.5.4
lru-cache: 6.0.0
prettier: ^2.7.1
ts-jest: ^27.1.4
tsup: ^6.5.0
Expand Down Expand Up @@ -24788,6 +24789,15 @@ __metadata:
languageName: node
linkType: hard

"lru-cache@npm:6.0.0, lru-cache@npm:^6.0.0":
version: 6.0.0
resolution: "lru-cache@npm:6.0.0"
dependencies:
yallist: ^4.0.0
checksum: f97f499f898f23e4585742138a22f22526254fdba6d75d41a1c2526b3b6cc5747ef59c5612ba7375f42aca4f8461950e925ba08c991ead0651b4918b7c978297
languageName: node
linkType: hard

"lru-cache@npm:^4.0.1, lru-cache@npm:^4.1.2, lru-cache@npm:^4.1.5":
version: 4.1.5
resolution: "lru-cache@npm:4.1.5"
Expand All @@ -24807,15 +24817,6 @@ __metadata:
languageName: node
linkType: hard

"lru-cache@npm:^6.0.0":
version: 6.0.0
resolution: "lru-cache@npm:6.0.0"
dependencies:
yallist: ^4.0.0
checksum: f97f499f898f23e4585742138a22f22526254fdba6d75d41a1c2526b3b6cc5747ef59c5612ba7375f42aca4f8461950e925ba08c991ead0651b4918b7c978297
languageName: node
linkType: hard

"lru-cache@npm:^7.14.1":
version: 7.18.3
resolution: "lru-cache@npm:7.18.3"
Expand Down

0 comments on commit a6b3766

Please sign in to comment.