Skip to content
This repository has been archived by the owner on Jan 28, 2021. It is now read-only.

Check projection aliases when assigned to index. #640

Merged
merged 2 commits into from
Apr 2, 2019
Merged

Check projection aliases when assigned to index. #640

merged 2 commits into from
Apr 2, 2019

Conversation

kuba--
Copy link
Contributor

@kuba-- kuba-- commented Mar 20, 2019

Signed-off-by: kuba-- kuba@sourced.tech
Fixes #639

Collect projection aliases and pass down to IndexByExpression

@kuba-- kuba-- requested a review from a team March 20, 2019 22:27
@kuba-- kuba-- added the bug Something isn't working label Mar 20, 2019
@@ -18,6 +18,7 @@ require (
github.com/mitchellh/hashstructure v1.0.0
github.com/oliveagle/jsonpath v0.0.0-20180606110733-2e52cf6e6852
github.com/opentracing/opentracing-go v1.0.2
github.com/pbnjay/memory v0.0.0-20190104145345-974d429e7ae4
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this is missed from the master

sql/index.go Outdated
expressions := make([]string, len(expr))
for i, e := range expr {
expressions[i] = e.String()
expressions := make(map[string]string)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why are we passing expr and aliases all the way here instead of just passing the expressions already normalized?

Instead of collecting a map[string]string of aliases, we could collect a map[string]sql.Expression and just pass the values of that map as expr. Then, no API would need to change.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We need to pass aliases from assignIndexes to getIndexes, because how we can combine them into map[string]sql.Expression in an obvious way?
Moreover in func getIndexes(e sql.Expression, a *Analyzer) we cast expression and pass left/right to getIndexes again or to IndexByExpression.
What I can do is to combine expressions and aliases before calling IndexByExpression , so instead of having IndexByExpression(db string, aliases map[string]string, expr ...Expression)
we will have IndexByExpression(db string, expressions map[string]string), so basically move outside the IndexByExpression:

	expressions := make(map[string]string)
	for _, e := range expr {
		if aliases != nil {
			expressions[e.String()] = aliases[e.String()]
		} else {
			expressions[e.String()] = ""
		}
	}

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What I mean is not having to change the api of IndexByExpression or anything like that so they don't need to know anything about aliases.

Instead, just pass the dealiased expressions.

If you would have to pass mytable.A and someAlias to IndexByExpression, instead pass mytable.A and foo(mytable.B), no need for any kind of mapping.

That way the functions that look for indexes don't need to know anything about aliases of any kind, just looking for those expressions.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ok, so I can normalize expressions and aliases and pass to IndexByExpression only expressions (or actually):
IndexByExpression(db string, expr ...string) Index
because we don't need whole Expression object inside IndexByExpression and we already have aliases as strings.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

But if you got the string of the expression, means you had the expression. Then instead I would pass the sql.Expression itself so the API does not need to change. The fact that we use the string is an implementation detail, it's better to keep passing sql.Expressions IMHO.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@erizocosmico - PTAL (rollback).

Signed-off-by: kuba-- <kuba@sourced.tech>
@kuba-- kuba-- requested a review from erizocosmico March 21, 2019 12:41
@@ -42,8 +42,17 @@ func assignIndexes(a *Analyzer, node sql.Node) (map[string]*indexLookup, error)
return true
}

aliases := make(map[string]sql.Expression)
if prj, ok := filter.Child.(*plan.Project); ok {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There are two cases that I think we're not handling here but can happen:

  • The alias was not defined in this project but in another one. Example:
Filter(a = 5)
|- Project(a, mytable.B) <- a here is still an alias
    |- Project(mytable.A as a, mytable.B) <-- it's in another project
  • Group Bys can also define aliases because they act like projects

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Are aliases local per query?
How about shadowing?
What if I have

SELECT a as foo from (
    SELECT b as foo from T
)

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Subqueries use their own indexes

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

But how about aliases? Here I have foo for a and b.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

foo projected from the subquery should have T as table, so it's not an alias. Besides, you stop trying to find an alias for foo as soon as you find one.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I added a recursive function to deeply inspect all aliases.
I think group by is not really a case here because we don't use indexes for group by.

Signed-off-by: kuba-- <kuba@sourced.tech>
@ajnavarro ajnavarro merged commit 201bcd6 into src-d:master Apr 2, 2019
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
bug Something isn't working
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants