-
Notifications
You must be signed in to change notification settings - Fork 1.8k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
chore(tesseract): Fix issue with FILTER_PARAMS and issue with multi-stage behavior of non multi-stage members #9211
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,4 @@ | ||
import { getEnv } from '@cubejs-backend/shared'; | ||
import { UserError } from '../../../src/compiler/UserError'; | ||
import { PostgresQuery } from '../../../src/adapter/PostgresQuery'; | ||
import { BigqueryQuery } from '../../../src/adapter/BigqueryQuery'; | ||
|
@@ -312,9 +313,12 @@ describe('SQL Generation', () => { | |
|
||
cube('visitor_checkins', { | ||
sql: \` | ||
select * from visitor_checkins WHERE | ||
\${FILTER_PARAMS.visitor_checkins.created_at.filter('created_at')} AND | ||
\${FILTER_GROUP(FILTER_PARAMS.visitor_checkins.created_at.filter("(created_at - INTERVAL '3 DAY')"), FILTER_PARAMS.visitor_checkins.source.filter('source'))} | ||
select visitor_checkins.* from visitor_checkins left join visitors on visitor_checkins.visitor_id = visitors.id WHERE | ||
\${FILTER_PARAMS.visitor_checkins.created_at.filter('visitor_checkins.created_at')} AND | ||
\${FILTER_GROUP(FILTER_PARAMS.visitor_checkins.created_at.filter("(visitor_checkins.created_at - INTERVAL '3 DAY')"), FILTER_PARAMS.visitor_checkins.source.filter('visitor_checkins.source'))} | ||
AND \${SECURITY_CONTEXT.source.filter('visitors.source')} AND | ||
\${SECURITY_CONTEXT.sourceArray.filter(sourceArray => \`visitors.source in (\${sourceArray.join(',')})\`)} | ||
|
||
\`, | ||
sql_alias: \`vc\`, | ||
|
||
|
@@ -567,6 +571,84 @@ describe('SQL Generation', () => { | |
}, | ||
} | ||
}); | ||
|
||
cube('rollingWindowDates', { | ||
sql: \` | ||
SELECT cast('2024-01-13' AS timestamp) as time UNION ALL | ||
SELECT cast('2024-02-13' AS timestamp) as time UNION ALL | ||
SELECT cast('2024-03-13' AS timestamp) as time UNION ALL | ||
SELECT cast('2024-04-13' AS timestamp) as time UNION ALL | ||
SELECT cast('2024-05-13' AS timestamp) as time UNION ALL | ||
SELECT cast('2024-06-13' AS timestamp) as time UNION ALL | ||
SELECT cast('2024-07-13' AS timestamp) as time UNION ALL | ||
SELECT cast('2024-08-13' AS timestamp) as time UNION ALL | ||
SELECT cast('2024-09-13' AS timestamp) as time UNION ALL | ||
SELECT cast('2024-10-13' AS timestamp) as time UNION ALL | ||
SELECT cast('2024-11-13' AS timestamp) as time UNION ALL | ||
SELECT cast('2024-12-13' AS timestamp) as time | ||
\`, | ||
|
||
dimensions: { | ||
time: { | ||
type: 'time', | ||
sql: 'time', | ||
primaryKey: true | ||
} | ||
} | ||
}); | ||
|
||
cube('rollingWindowTest', { | ||
sql: \` | ||
SELECT 1 AS revenue, cast('2024-01-01' AS timestamp) as time UNION ALL | ||
SELECT 1 AS revenue, cast('2024-02-01' AS timestamp) as time UNION ALL | ||
SELECT 1 AS revenue, cast('2024-03-01' AS timestamp) as time UNION ALL | ||
SELECT 1 AS revenue, cast('2024-04-01' AS timestamp) as time UNION ALL | ||
SELECT 1 AS revenue, cast('2024-05-01' AS timestamp) as time UNION ALL | ||
SELECT 1 AS revenue, cast('2024-06-01' AS timestamp) as time UNION ALL | ||
SELECT 1 AS revenue, cast('2024-07-01' AS timestamp) as time UNION ALL | ||
SELECT 1 AS revenue, cast('2024-08-01' AS timestamp) as time UNION ALL | ||
SELECT 1 AS revenue, cast('2024-09-01' AS timestamp) as time UNION ALL | ||
SELECT 1 AS revenue, cast('2024-10-01' AS timestamp) as time UNION ALL | ||
SELECT 1 AS revenue, cast('2024-11-01' AS timestamp) as time UNION ALL | ||
SELECT 1 AS revenue, cast('2024-12-01' AS timestamp) as time | ||
\`, | ||
|
||
dimensions: { | ||
time: { | ||
type: 'time', | ||
sql: 'time', | ||
primaryKey: true | ||
} | ||
}, | ||
measures: { | ||
revenue: { | ||
sql: 'revenue', | ||
type: 'sum', | ||
filters: [{ | ||
sql: \`\${rollingWindowDates.time} <= current_date\` | ||
}] | ||
}, | ||
revenue_ytd: { | ||
sql: \`\${CUBE.revenue}\`, | ||
type: 'sum', | ||
rolling_window: { | ||
type: 'to_date', | ||
granularity: 'year' | ||
} | ||
}, | ||
revenue_ms: { | ||
sql: \`\${CUBE.revenue}\`, | ||
type: 'sum', | ||
multi_stage: true, | ||
}, | ||
}, | ||
joins: { | ||
rollingWindowDates: { | ||
relationship: 'manyToOne', | ||
sql: \`\${CUBE}.time = date_trunc('month', \${rollingWindowDates.time})\` | ||
} | ||
} | ||
}); | ||
`); | ||
|
||
it('simple join', async () => { | ||
|
@@ -1966,7 +2048,7 @@ describe('SQL Generation', () => { | |
])); | ||
|
||
it( | ||
'contains filter 1', | ||
'contains filter', | ||
() => runQueryTest({ | ||
measures: [], | ||
dimensions: [ | ||
|
@@ -2761,7 +2843,7 @@ describe('SQL Generation', () => { | |
] | ||
)); | ||
|
||
it('rank measure 1', async () => runQueryTest( | ||
it('rank measure', async () => runQueryTest( | ||
{ | ||
measures: ['visitors.revenue_rank'], | ||
}, | ||
|
@@ -3152,6 +3234,37 @@ describe('SQL Generation', () => { | |
}] | ||
)); | ||
|
||
if (getEnv('nativeSqlPlanner')) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think it would be more correct to mark the test as skipped in case of not using |
||
it('nested aggregations with filtered measures and rolling windows', async () => runQueryTest( | ||
{ | ||
measures: ['rollingWindowTest.revenue_ms'], | ||
}, | ||
[{ | ||
rolling_window_test__revenue_ms: '12' | ||
}] | ||
)); | ||
} | ||
|
||
it('multiplied sum and count no dimensions through view', async () => runQueryTest( | ||
{ | ||
measures: ['visitors_visitors_checkins_view.revenue', 'visitors_visitors_checkins_view.visitor_checkins_count'], | ||
}, | ||
[{ | ||
visitors_visitors_checkins_view__revenue: '2000', | ||
visitors_visitors_checkins_view__visitor_checkins_count: '6' | ||
}] | ||
)); | ||
|
||
it('multiplied sum no dimensions through view', async () => runQueryTest( | ||
{ | ||
measures: ['visitors_visitors_checkins_view.revenue', 'visitors_visitors_checkins_view.id_sum'], | ||
}, | ||
[{ | ||
visitors_visitors_checkins_view__revenue: '2000', | ||
visitors_visitors_checkins_view__id_sum: '21' | ||
}] | ||
)); | ||
|
||
// Subquery aggregation for multiplied measure (and any `keysSelect` for that matter) | ||
// should pick up all dimensions, even through member expressions | ||
it('multiplied sum with dimension member expressions', async () => runQueryTest( | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -138,7 +138,7 @@ impl<IT: InnerTypes> ser::Serializer for NativeSerdeSerializer<IT> { | |
} | ||
|
||
fn serialize_none(self) -> Result<Self::Ok, Self::Error> { | ||
Ok(self.context.undefined()?) | ||
Ok(self.context.null()?) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Are you sure There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yes, I think it's justified here. In general, this change was made to normally pass parameters in filters, which can be either a string or null. |
||
} | ||
|
||
fn serialize_some<T>(self, value: &T) -> Result<Self::Ok, Self::Error> | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can you explain the change? Could you tell me why you didn't create another cube/test or why this one doesn't suit your needs?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is an fix of an incorrect data model for tests.
SECURITY_CONTEXT.filter
should work regardless of whether the cube has built a join for a particular query or not. Therefore, this filter must be written directly in the sql of the cube. This is one of the reasons why SECURITY_CONTEXT is deprecated. In the case of tesseract it led to the fact that in one of the tests fails, because tesseract knows how to omit unnecessary joins in subqueries. But this is not the error of tesseract, but of incorrect use of SECURITY_CONTEXT in the model.