Skip to content

Commit

Permalink
Optimize Duplications of ComputedField queries
Browse files Browse the repository at this point in the history
For each modelName the a `ComputedField` is only compiled into abstract-sql-query once. Afterwards it's used as ReferencedField

Change-type: minor
Signed-off-by: fisehara <[email protected]>
  • Loading branch information
fisehara committed Jun 28, 2023
1 parent 298161f commit 0229bd5
Show file tree
Hide file tree
Showing 5 changed files with 235 additions and 10 deletions.
41 changes: 32 additions & 9 deletions src/odata-to-abstract-sql.ts
Original file line number Diff line number Diff line change
Expand Up @@ -117,6 +117,8 @@ type Overwrite<T, U> = Pick<T, Exclude<keyof T, keyof U>> & U;
type RequiredField<T, F extends keyof T> = Overwrite<T, Required<Pick<T, F>>>;
type AliasedResource = RequiredField<Resource, 'tableAlias'>;

type AlreadyComputedFieldsLookup = { [resourceAndFieldName: string]: boolean };

export type ResourceFunction = (
this: OData2AbstractSQL,
property: {
Expand Down Expand Up @@ -298,10 +300,17 @@ export const isBindReference = (maybeBind: {
);
};

const isDynamicResource = (resource: Resource): boolean => {
const isDynamicResource = (
resource: Resource,
alreadyComputedFieldsLookup: AlreadyComputedFieldsLookup,
): boolean => {
return (
resource.definition != null ||
resource.fields.some((f) => f.computed != null)
resource.fields.some(
(f) =>
f.computed != null &&
!alreadyComputedFieldsLookup[resource.name + '$' + f.fieldName],
)
);
};

Expand All @@ -327,6 +336,7 @@ export class OData2AbstractSQL {
public defaultResource: Resource | undefined;
public bindVarsLength: number = 0;
private checkAlias: (alias: string) => string;
private alreadyComputedFields: AlreadyComputedFieldsLookup = {};

constructor(
private clientModel: RequiredAbstractSqlModelSubset,
Expand Down Expand Up @@ -543,7 +553,9 @@ export class OData2AbstractSQL {
// For updates/deletes that we use a `WHERE id IN (SELECT...)` subquery to apply options and in the case of a definition
// we make sure to always apply it. This means that the definition will still be applied for these queries
if (
(hasQueryOpts || isDynamicResource(resource) || pathKeyWhere != null) &&
(hasQueryOpts ||
isDynamicResource(resource, this.alreadyComputedFields) ||
pathKeyWhere != null) &&
(method === 'POST' || method === 'PUT-INSERT')
) {
// For insert statements we need to use an INSERT INTO ... SELECT * FROM (binds) WHERE ... style query
Expand All @@ -562,8 +574,9 @@ export class OData2AbstractSQL {
'SelectQuery',
[
'Select',
(resource.modifyFields ?? resource.fields).map(
(field): AliasNode<CastNode> => {
(resource.modifyFields ?? resource.fields)
.filter((field) => field.computed == null)
.map((field): AliasNode<CastNode> => {
const alias = field.fieldName;
const bindVar = bindVars?.find((v) => v[0] === alias);
const value = bindVar?.[1] ?? ['Null'];
Expand All @@ -573,8 +586,7 @@ export class OData2AbstractSQL {
);
}
return ['Alias', ['Cast', value, field.dataType], alias];
},
),
}),
],
],
'$insert',
Expand Down Expand Up @@ -687,7 +699,8 @@ export class OData2AbstractSQL {
// we make sure to always apply it. This means that the definition will still be applied for these queries, for insert queries
// this is handled when we set the 'Values'
if (
(hasQueryOpts || isDynamicResource(resource)) &&
(hasQueryOpts ||
isDynamicResource(resource, this.alreadyComputedFields)) &&
(method === 'PUT' ||
method === 'PATCH' ||
method === 'MERGE' ||
Expand Down Expand Up @@ -991,11 +1004,18 @@ export class OData2AbstractSQL {
fieldName: string,
computed?: AbstractSqlQuery,
alias: string = fieldName,
// Setting this flag to true will ignore the lookup for alreadyComputedFields
// and will compile the computed Field statement into the abstract SQL statement regardless
forceCompilingComputedField: boolean = false,
):
| ReferencedFieldNode
| AliasNode<ReferencedFieldNode>
| AliasNode<AbstractSqlQuery> {
if (computed) {
const key = resource.name + '$' + fieldName;
if (
computed &&
(!this.alreadyComputedFields[key] || forceCompilingComputedField)
) {
if (
resource.tableAlias != null &&
resource.tableAlias !== resource.name
Expand All @@ -1006,6 +1026,7 @@ export class OData2AbstractSQL {
resource.tableAlias,
);
}
this.alreadyComputedFields[key] = true;
return ['Alias', computed, alias];
}
const referencedField = this.ReferencedField(resource, fieldName);
Expand Down Expand Up @@ -1632,6 +1653,7 @@ export class OData2AbstractSQL {
this.putReset();
this.extraBodyVars = {};
this.extraBindVars = [] as unknown as ODataBinds;
this.alreadyComputedFields = {};
}

putReset() {
Expand Down Expand Up @@ -1695,6 +1717,7 @@ export class OData2AbstractSQL {
sqlNameToODataName(field.fieldName),
field.computed,
field.fieldName,
true,
),
),
];
Expand Down
24 changes: 24 additions & 0 deletions test/chai-sql.js
Original file line number Diff line number Diff line change
Expand Up @@ -105,6 +105,20 @@ const generateClientModel = function (input) {
const sbvrModel = fs.readFileSync(require.resolve('./model.sbvr'), 'utf8');
export const clientModel = generateClientModel(sbvrModel);

clientModel.tables['copilot'].fields.push({
fieldName: 'is blocked',
dataType: 'Boolean',
// The cast is needed because AbstractSqlQuery cannot express a constant value.
computed: ['Boolean', false],
});

clientModel.tables['copilot'].fields.push({
fieldName: 'rank',
dataType: 'Text',
// The cast is needed because AbstractSqlQuery cannot express a constant value.
computed: ['Text', 'Junior'],
});

const odataNameToSqlName = (odataName) =>
odataName.replace(/__/g, '-').replace(/_/g, ' ');

Expand Down Expand Up @@ -360,3 +374,13 @@ export let teamFields = [
];

export let $count = [['Alias', ['Count', '*'], '$count']];

export let copilotFields = [
['Alias', ['ReferencedField', 'copilot', 'created at'], 'created_at'],
['Alias', ['ReferencedField', 'copilot', 'modified at'], 'modified_at'],
['ReferencedField', 'copilot', 'id'],
['ReferencedField', 'copilot', 'person'],
['ReferencedField', 'copilot', 'assists-pilot'],
['Alias', ['ReferencedField', 'copilot', 'is blocked'], 'is_blocked'],
['ReferencedField', 'copilot', 'rank'],
];
94 changes: 94 additions & 0 deletions test/filterby.js
Original file line number Diff line number Diff line change
Expand Up @@ -1433,3 +1433,97 @@ run(function () {
}),
);
});

test(`/copilot?$select=id,rank&$filter=rank eq 'major'`, (result) =>
it(`should get and filter copilot on computed field rank`, () => {
expect(result).to.be.a.query.to.deep.equal([
'SelectQuery',
[
'Select',
[
['ReferencedField', 'copilot', 'id'],
['ReferencedField', 'copilot', 'rank'],
],
],
[
'From',
[
'Alias',
[
'SelectQuery',
[
'Select',
[
['Field', '*'],
['Alias', ['Boolean', false], 'is blocked'],
['Alias', ['Text', 'Junior'], 'rank'],
],
],
['From', ['Table', 'copilot']],
],
'copilot',
],
],
[
'Where',
[
'IsNotDistinctFrom',
['ReferencedField', 'copilot', 'rank'],
['Bind', 0],
],
],
]);
}));

test(
`/copilot?$select=id,rank&$filter=rank eq 'major'`,
'PATCH',
{ assists__pilot: 1 },
(result) =>
it(`should PATCH copilot based on filtered computed field rank`, () => {
expect(result).to.be.a.query.to.deep.equal([
'UpdateQuery',
['From', ['Table', 'copilot']],
[
'Where',
[
'In',
['ReferencedField', 'copilot', 'id'],
[
'SelectQuery',
['Select', [['ReferencedField', 'copilot', 'id']]],
[
'From',
[
'Alias',
[
'SelectQuery',
[
'Select',
[
['Field', '*'],
['Alias', ['Boolean', false], 'is blocked'],
['Alias', ['Text', 'Junior'], 'rank'],
],
],
['From', ['Table', 'copilot']],
],
'copilot',
],
],
[
'Where',
[
'IsNotDistinctFrom',
['ReferencedField', 'copilot', 'rank'],
['Bind', 0],
],
],
],
],
],
['Fields', ['assists-pilot']],
['Values', [['Bind', 'copilot', 'assists__pilot']]],
]);
}),
);
8 changes: 7 additions & 1 deletion test/model.sbvr
Original file line number Diff line number Diff line change
Expand Up @@ -57,4 +57,10 @@ Fact type: pilot has hire date

Fact type: pilot1 was trained by pilot2
Synonymous Form: pilot2 trained pilot1
Necessity: each pilot was trained by exactly one pilot
Necessity: each pilot was trained by exactly one pilot

Term: copilot
Concept Type: pilot

Fact Type: copilot assists pilot
Necessity: each copilot assists at most one pilot
78 changes: 78 additions & 0 deletions test/select.js
Original file line number Diff line number Diff line change
Expand Up @@ -152,3 +152,81 @@ test('/pilot?$select=can_fly__plane/plane/id', (result) =>
['ReferencedField', 'pilot.pilot-can fly-plane.plane', 'id'],
],
])));

test('/copilot?$select=*', (result) =>
it('should select * from copilot', () =>
expect(result).to.be.a.query.to.deep.equal([
'SelectQuery',
[
'Select',
[
['Alias', ['ReferencedField', 'copilot', 'created at'], 'created_at'],
[
'Alias',
['ReferencedField', 'copilot', 'modified at'],
'modified_at',
],
['ReferencedField', 'copilot', 'id'],
['ReferencedField', 'copilot', 'pilot'],
[
'Alias',
['ReferencedField', 'copilot', 'assists-pilot'],
'assists__pilot',
],
['Alias', ['ReferencedField', 'copilot', 'is blocked'], 'is_blocked'],
['ReferencedField', 'copilot', 'rank'],
],
],
[
'From',
[
'Alias',
[
'SelectQuery',
[
'Select',
[
['Field', '*'],
['Alias', ['Boolean', false], 'is blocked'],
['Alias', ['Text', 'Junior'], 'rank'],
],
],
['From', ['Table', 'copilot']],
],
'copilot',
],
],
])));

test('/copilot?$select=id,is_blocked,rank', (result) =>
it('should select * from copilot', () =>
expect(result).to.be.a.query.to.deep.equal([
'SelectQuery',
[
'Select',
[
['ReferencedField', 'copilot', 'id'],
['Alias', ['ReferencedField', 'copilot', 'is blocked'], 'is_blocked'],
['ReferencedField', 'copilot', 'rank'],
],
],
[
'From',
[
'Alias',
[
'SelectQuery',
[
'Select',
[
['Field', '*'],
['Alias', ['Boolean', false], 'is blocked'],
['Alias', ['Text', 'Junior'], 'rank'],
],
],
['From', ['Table', 'copilot']],
],
'copilot',
],
],
])));

0 comments on commit 0229bd5

Please sign in to comment.