| 
									
										
										
										
											2016-04-13 22:08:22 +08:00
										 |  |  | // Copyright 2016 The Prometheus Authors
 | 
					
						
							|  |  |  | // Licensed under the Apache License, Version 2.0 (the "License");
 | 
					
						
							|  |  |  | // you may not use this file except in compliance with the License.
 | 
					
						
							|  |  |  | // You may obtain a copy of the License at
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //     http://www.apache.org/licenses/LICENSE-2.0
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // Unless required by applicable law or agreed to in writing, software
 | 
					
						
							|  |  |  | // distributed under the License is distributed on an "AS IS" BASIS,
 | 
					
						
							|  |  |  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
					
						
							|  |  |  | // See the License for the specific language governing permissions and
 | 
					
						
							|  |  |  | // limitations under the License.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | package promql | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2017-10-05 03:04:15 +08:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2019-10-10 08:06:53 +08:00
										 |  |  | 	"errors" | 
					
						
							| 
									
										
										
										
											2020-01-28 06:29:44 +08:00
										 |  |  | 	"io/ioutil" | 
					
						
							|  |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2020-01-17 19:21:44 +08:00
										 |  |  | 	"sort" | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	"testing" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-12 02:45:52 +08:00
										 |  |  | 	"github.com/go-kit/kit/log" | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	"github.com/stretchr/testify/require" | 
					
						
							| 
									
										
										
										
											2020-10-22 17:00:08 +08:00
										 |  |  | 	"go.uber.org/goleak" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-18 21:51:10 +08:00
										 |  |  | 	"github.com/prometheus/prometheus/pkg/labels" | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 	"github.com/prometheus/prometheus/pkg/timestamp" | 
					
						
							| 
									
										
										
										
											2020-02-04 00:32:23 +08:00
										 |  |  | 	"github.com/prometheus/prometheus/promql/parser" | 
					
						
							| 
									
										
										
										
											2018-02-15 20:08:00 +08:00
										 |  |  | 	"github.com/prometheus/prometheus/storage" | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-24 17:10:20 +08:00
										 |  |  | func TestMain(m *testing.M) { | 
					
						
							|  |  |  | 	goleak.VerifyTestMain(m) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-23 17:06:51 +08:00
										 |  |  | func TestQueryConcurrency(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2020-01-28 06:29:44 +08:00
										 |  |  | 	maxConcurrency := 10 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dir, err := ioutil.TempDir("", "test_concurrency") | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.NoError(t, err) | 
					
						
							| 
									
										
										
										
											2020-01-28 06:29:44 +08:00
										 |  |  | 	defer os.RemoveAll(dir) | 
					
						
							|  |  |  | 	queryTracker := NewActiveQueryTracker(dir, maxConcurrency, nil) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 	opts := EngineOpts{ | 
					
						
							| 
									
										
										
										
											2020-01-28 06:29:44 +08:00
										 |  |  | 		Logger:             nil, | 
					
						
							|  |  |  | 		Reg:                nil, | 
					
						
							|  |  |  | 		MaxSamples:         10, | 
					
						
							|  |  |  | 		Timeout:            100 * time.Second, | 
					
						
							|  |  |  | 		ActiveQueryTracker: queryTracker, | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-01-10 00:44:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 	engine := NewEngine(opts) | 
					
						
							| 
									
										
										
										
											2016-09-16 06:58:06 +08:00
										 |  |  | 	ctx, cancelCtx := context.WithCancel(context.Background()) | 
					
						
							|  |  |  | 	defer cancelCtx() | 
					
						
							| 
									
										
										
										
											2015-05-01 06:49:19 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	block := make(chan struct{}) | 
					
						
							|  |  |  | 	processing := make(chan struct{}) | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	f := func(context.Context) error { | 
					
						
							| 
									
										
										
										
											2015-05-01 06:49:19 +08:00
										 |  |  | 		processing <- struct{}{} | 
					
						
							|  |  |  | 		<-block | 
					
						
							|  |  |  | 		return nil | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-05-01 06:49:19 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-29 04:38:49 +08:00
										 |  |  | 	for i := 0; i < maxConcurrency; i++ { | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 		q := engine.newTestQuery(f) | 
					
						
							| 
									
										
											  
											
												promql: Allow per-query contexts.
For Weaveworks' Frankenstein, we need to support multitenancy. In
Frankenstein, we initially solved this without modifying the promql
package at all: we constructed a new promql.Engine for every
query and injected a storage implementation into that engine which would
be primed to only collect data for a given user.
This is problematic to upstream, however. Prometheus assumes that there
is only one engine: the query concurrency gate is part of the engine,
and the engine contains one central cancellable context to shut down all
queries. Also, creating a new engine for every query seems like overkill.
Thus, we want to be able to pass per-query contexts into a single engine.
This change gets rid of the promql.Engine's built-in base context and
allows passing in a per-query context instead. Central cancellation of
all queries is still possible by deriving all passed-in contexts from
one central one, but this is now the responsibility of the caller. The
central query context is now created in main() and passed into the
relevant components (web handler / API, rule manager).
In a next step, the per-query context would have to be passed to the
storage implementation, so that the storage can implement multi-tenancy
or other features based on the contextual information.
											
										 
											2016-09-15 19:52:50 +08:00
										 |  |  | 		go q.Exec(ctx) | 
					
						
							| 
									
										
										
										
											2015-05-01 06:49:19 +08:00
										 |  |  | 		select { | 
					
						
							|  |  |  | 		case <-processing: | 
					
						
							|  |  |  | 			// Expected.
 | 
					
						
							| 
									
										
										
										
											2015-09-23 17:06:51 +08:00
										 |  |  | 		case <-time.After(20 * time.Millisecond): | 
					
						
							| 
									
										
										
										
											2015-05-01 06:49:19 +08:00
										 |  |  | 			t.Fatalf("Query within concurrency threshold not being executed") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 	q := engine.newTestQuery(f) | 
					
						
							| 
									
										
											  
											
												promql: Allow per-query contexts.
For Weaveworks' Frankenstein, we need to support multitenancy. In
Frankenstein, we initially solved this without modifying the promql
package at all: we constructed a new promql.Engine for every
query and injected a storage implementation into that engine which would
be primed to only collect data for a given user.
This is problematic to upstream, however. Prometheus assumes that there
is only one engine: the query concurrency gate is part of the engine,
and the engine contains one central cancellable context to shut down all
queries. Also, creating a new engine for every query seems like overkill.
Thus, we want to be able to pass per-query contexts into a single engine.
This change gets rid of the promql.Engine's built-in base context and
allows passing in a per-query context instead. Central cancellation of
all queries is still possible by deriving all passed-in contexts from
one central one, but this is now the responsibility of the caller. The
central query context is now created in main() and passed into the
relevant components (web handler / API, rule manager).
In a next step, the per-query context would have to be passed to the
storage implementation, so that the storage can implement multi-tenancy
or other features based on the contextual information.
											
										 
											2016-09-15 19:52:50 +08:00
										 |  |  | 	go q.Exec(ctx) | 
					
						
							| 
									
										
										
										
											2015-05-01 06:49:19 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	select { | 
					
						
							|  |  |  | 	case <-processing: | 
					
						
							| 
									
										
										
										
											2016-07-12 02:27:25 +08:00
										 |  |  | 		t.Fatalf("Query above concurrency threshold being executed") | 
					
						
							| 
									
										
										
										
											2015-09-23 17:06:51 +08:00
										 |  |  | 	case <-time.After(20 * time.Millisecond): | 
					
						
							| 
									
										
										
										
											2015-05-01 06:49:19 +08:00
										 |  |  | 		// Expected.
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Terminate a running query.
 | 
					
						
							|  |  |  | 	block <- struct{}{} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	select { | 
					
						
							|  |  |  | 	case <-processing: | 
					
						
							|  |  |  | 		// Expected.
 | 
					
						
							| 
									
										
										
										
											2015-09-23 17:06:51 +08:00
										 |  |  | 	case <-time.After(20 * time.Millisecond): | 
					
						
							| 
									
										
										
										
											2015-05-01 06:49:19 +08:00
										 |  |  | 		t.Fatalf("Query within concurrency threshold not being executed") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Terminate remaining queries.
 | 
					
						
							| 
									
										
										
										
											2020-01-29 04:38:49 +08:00
										 |  |  | 	for i := 0; i < maxConcurrency; i++ { | 
					
						
							| 
									
										
										
										
											2015-05-01 06:49:19 +08:00
										 |  |  | 		block <- struct{}{} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | func TestQueryTimeout(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 	opts := EngineOpts{ | 
					
						
							| 
									
										
										
										
											2020-01-29 04:38:49 +08:00
										 |  |  | 		Logger:     nil, | 
					
						
							|  |  |  | 		Reg:        nil, | 
					
						
							|  |  |  | 		MaxSamples: 10, | 
					
						
							|  |  |  | 		Timeout:    5 * time.Millisecond, | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	engine := NewEngine(opts) | 
					
						
							| 
									
										
										
										
											2016-09-16 06:58:06 +08:00
										 |  |  | 	ctx, cancelCtx := context.WithCancel(context.Background()) | 
					
						
							|  |  |  | 	defer cancelCtx() | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 	query := engine.newTestQuery(func(ctx context.Context) error { | 
					
						
							| 
									
										
										
										
											2020-01-22 18:09:31 +08:00
										 |  |  | 		time.Sleep(100 * time.Millisecond) | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 		return contextDone(ctx, "test statement execution") | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												promql: Allow per-query contexts.
For Weaveworks' Frankenstein, we need to support multitenancy. In
Frankenstein, we initially solved this without modifying the promql
package at all: we constructed a new promql.Engine for every
query and injected a storage implementation into that engine which would
be primed to only collect data for a given user.
This is problematic to upstream, however. Prometheus assumes that there
is only one engine: the query concurrency gate is part of the engine,
and the engine contains one central cancellable context to shut down all
queries. Also, creating a new engine for every query seems like overkill.
Thus, we want to be able to pass per-query contexts into a single engine.
This change gets rid of the promql.Engine's built-in base context and
allows passing in a per-query context instead. Central cancellation of
all queries is still possible by deriving all passed-in contexts from
one central one, but this is now the responsibility of the caller. The
central query context is now created in main() and passed into the
relevant components (web handler / API, rule manager).
In a next step, the per-query context would have to be passed to the
storage implementation, so that the storage can implement multi-tenancy
or other features based on the contextual information.
											
										 
											2016-09-15 19:52:50 +08:00
										 |  |  | 	res := query.Exec(ctx) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.Error(t, res.Err, "expected timeout error but got none") | 
					
						
							| 
									
										
										
										
											2019-10-10 08:06:53 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	var e ErrQueryTimeout | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.True(t, errors.As(res.Err, &e), "expected timeout error but got: %s", res.Err) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-10 08:06:53 +08:00
										 |  |  | const errQueryCanceled = ErrQueryCanceled("test statement execution") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | func TestQueryCancel(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 	opts := EngineOpts{ | 
					
						
							| 
									
										
										
										
											2020-01-29 04:38:49 +08:00
										 |  |  | 		Logger:     nil, | 
					
						
							|  |  |  | 		Reg:        nil, | 
					
						
							|  |  |  | 		MaxSamples: 10, | 
					
						
							|  |  |  | 		Timeout:    10 * time.Second, | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	engine := NewEngine(opts) | 
					
						
							| 
									
										
										
										
											2016-09-16 06:58:06 +08:00
										 |  |  | 	ctx, cancelCtx := context.WithCancel(context.Background()) | 
					
						
							|  |  |  | 	defer cancelCtx() | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 	// Cancel a running query before it completes.
 | 
					
						
							|  |  |  | 	block := make(chan struct{}) | 
					
						
							|  |  |  | 	processing := make(chan struct{}) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 	query1 := engine.newTestQuery(func(ctx context.Context) error { | 
					
						
							|  |  |  | 		processing <- struct{}{} | 
					
						
							|  |  |  | 		<-block | 
					
						
							|  |  |  | 		return contextDone(ctx, "test statement execution") | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2015-04-29 17:08:56 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	var res *Result | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	go func() { | 
					
						
							| 
									
										
											  
											
												promql: Allow per-query contexts.
For Weaveworks' Frankenstein, we need to support multitenancy. In
Frankenstein, we initially solved this without modifying the promql
package at all: we constructed a new promql.Engine for every
query and injected a storage implementation into that engine which would
be primed to only collect data for a given user.
This is problematic to upstream, however. Prometheus assumes that there
is only one engine: the query concurrency gate is part of the engine,
and the engine contains one central cancellable context to shut down all
queries. Also, creating a new engine for every query seems like overkill.
Thus, we want to be able to pass per-query contexts into a single engine.
This change gets rid of the promql.Engine's built-in base context and
allows passing in a per-query context instead. Central cancellation of
all queries is still possible by deriving all passed-in contexts from
one central one, but this is now the responsibility of the caller. The
central query context is now created in main() and passed into the
relevant components (web handler / API, rule manager).
In a next step, the per-query context would have to be passed to the
storage implementation, so that the storage can implement multi-tenancy
or other features based on the contextual information.
											
										 
											2016-09-15 19:52:50 +08:00
										 |  |  | 		res = query1.Exec(ctx) | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 		processing <- struct{}{} | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	}() | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	<-processing | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	query1.Cancel() | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 	block <- struct{}{} | 
					
						
							|  |  |  | 	<-processing | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.Error(t, res.Err, "expected cancellation error for query1 but got none") | 
					
						
							|  |  |  | 	require.Equal(t, errQueryCanceled, res.Err) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 	// Canceling a query before starting it must have no effect.
 | 
					
						
							|  |  |  | 	query2 := engine.newTestQuery(func(ctx context.Context) error { | 
					
						
							|  |  |  | 		return contextDone(ctx, "test statement execution") | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	query2.Cancel() | 
					
						
							| 
									
										
											  
											
												promql: Allow per-query contexts.
For Weaveworks' Frankenstein, we need to support multitenancy. In
Frankenstein, we initially solved this without modifying the promql
package at all: we constructed a new promql.Engine for every
query and injected a storage implementation into that engine which would
be primed to only collect data for a given user.
This is problematic to upstream, however. Prometheus assumes that there
is only one engine: the query concurrency gate is part of the engine,
and the engine contains one central cancellable context to shut down all
queries. Also, creating a new engine for every query seems like overkill.
Thus, we want to be able to pass per-query contexts into a single engine.
This change gets rid of the promql.Engine's built-in base context and
allows passing in a per-query context instead. Central cancellation of
all queries is still possible by deriving all passed-in contexts from
one central one, but this is now the responsibility of the caller. The
central query context is now created in main() and passed into the
relevant components (web handler / API, rule manager).
In a next step, the per-query context would have to be passed to the
storage implementation, so that the storage can implement multi-tenancy
or other features based on the contextual information.
											
										 
											2016-09-15 19:52:50 +08:00
										 |  |  | 	res = query2.Exec(ctx) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.NoError(t, res.Err) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-15 20:08:00 +08:00
										 |  |  | // errQuerier implements storage.Querier which always returns error.
 | 
					
						
							|  |  |  | type errQuerier struct { | 
					
						
							|  |  |  | 	err error | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-10 00:57:31 +08:00
										 |  |  | func (q *errQuerier) Select(bool, *storage.SelectHints, ...*labels.Matcher) storage.SeriesSet { | 
					
						
							|  |  |  | 	return errSeriesSet{err: q.err} | 
					
						
							| 
									
										
										
										
											2018-02-15 20:08:00 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-02-10 01:38:35 +08:00
										 |  |  | func (*errQuerier) LabelValues(string, ...*labels.Matcher) ([]string, storage.Warnings, error) { | 
					
						
							|  |  |  | 	return nil, nil, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | func (*errQuerier) LabelNames() ([]string, storage.Warnings, error) { return nil, nil, nil } | 
					
						
							|  |  |  | func (*errQuerier) Close() error                                    { return nil } | 
					
						
							| 
									
										
										
										
											2018-02-15 20:08:00 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // errSeriesSet implements storage.SeriesSet which always returns error.
 | 
					
						
							|  |  |  | type errSeriesSet struct { | 
					
						
							|  |  |  | 	err error | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-10 00:57:31 +08:00
										 |  |  | func (errSeriesSet) Next() bool                   { return false } | 
					
						
							|  |  |  | func (errSeriesSet) At() storage.Series           { return nil } | 
					
						
							|  |  |  | func (e errSeriesSet) Err() error                 { return e.err } | 
					
						
							|  |  |  | func (e errSeriesSet) Warnings() storage.Warnings { return nil } | 
					
						
							| 
									
										
										
										
											2018-02-15 20:08:00 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | func TestQueryError(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 	opts := EngineOpts{ | 
					
						
							| 
									
										
										
										
											2020-01-29 04:38:49 +08:00
										 |  |  | 		Logger:     nil, | 
					
						
							|  |  |  | 		Reg:        nil, | 
					
						
							|  |  |  | 		MaxSamples: 10, | 
					
						
							|  |  |  | 		Timeout:    10 * time.Second, | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	engine := NewEngine(opts) | 
					
						
							| 
									
										
										
										
											2019-03-26 07:01:12 +08:00
										 |  |  | 	errStorage := ErrStorage{errors.New("storage error")} | 
					
						
							| 
									
										
										
										
											2018-02-15 20:08:00 +08:00
										 |  |  | 	queryable := storage.QueryableFunc(func(ctx context.Context, mint, maxt int64) (storage.Querier, error) { | 
					
						
							|  |  |  | 		return &errQuerier{err: errStorage}, nil | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	ctx, cancelCtx := context.WithCancel(context.Background()) | 
					
						
							|  |  |  | 	defer cancelCtx() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	vectorQuery, err := engine.NewInstantQuery(queryable, "foo", time.Unix(1, 0)) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.NoError(t, err) | 
					
						
							| 
									
										
										
										
											2019-10-10 08:06:53 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-15 20:08:00 +08:00
										 |  |  | 	res := vectorQuery.Exec(ctx) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.Error(t, res.Err, "expected error on failed select but got none") | 
					
						
							|  |  |  | 	require.True(t, errors.Is(res.Err, errStorage), "expected error doesn't match") | 
					
						
							| 
									
										
										
										
											2018-02-15 20:08:00 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	matrixQuery, err := engine.NewInstantQuery(queryable, "foo[1m]", time.Unix(1, 0)) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.NoError(t, err) | 
					
						
							| 
									
										
										
										
											2019-10-10 08:06:53 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-15 20:08:00 +08:00
										 |  |  | 	res = matrixQuery.Exec(ctx) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.Error(t, res.Err, "expected error on failed select but got none") | 
					
						
							|  |  |  | 	require.True(t, errors.Is(res.Err, errStorage), "expected error doesn't match") | 
					
						
							| 
									
										
										
										
											2018-02-15 20:08:00 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | type noopHintRecordingQueryable struct { | 
					
						
							|  |  |  | 	hints []*storage.SelectHints | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | func (h *noopHintRecordingQueryable) Querier(context.Context, int64, int64) (storage.Querier, error) { | 
					
						
							|  |  |  | 	return &hintRecordingQuerier{Querier: &errQuerier{}, h: h}, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type hintRecordingQuerier struct { | 
					
						
							|  |  |  | 	storage.Querier | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 	h *noopHintRecordingQueryable | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | func (h *hintRecordingQuerier) Select(sortSeries bool, hints *storage.SelectHints, matchers ...*labels.Matcher) storage.SeriesSet { | 
					
						
							|  |  |  | 	h.h.hints = append(h.h.hints, hints) | 
					
						
							|  |  |  | 	return h.Querier.Select(sortSeries, hints, matchers...) | 
					
						
							| 
									
										
										
										
											2019-06-17 15:31:17 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | func TestSelectHintsSetCorrectly(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	opts := EngineOpts{ | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 		Logger:           nil, | 
					
						
							|  |  |  | 		Reg:              nil, | 
					
						
							|  |  |  | 		MaxSamples:       10, | 
					
						
							|  |  |  | 		Timeout:          10 * time.Second, | 
					
						
							|  |  |  | 		LookbackDelta:    5 * time.Second, | 
					
						
							|  |  |  | 		EnableAtModifier: true, | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 	for _, tc := range []struct { | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 		query string | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		// All times are in milliseconds.
 | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 		start int64 | 
					
						
							|  |  |  | 		end   int64 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		// TODO(bwplotka): Add support for better hints when subquerying.
 | 
					
						
							|  |  |  | 		expected []*storage.SelectHints | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	}{{ | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "foo", start: 10000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 5000, End: 10000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "foo @ 15", start: 10000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 10000, End: 15000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "foo @ 1", start: 10000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: -4000, End: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "foo[2m]", start: 200000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 80000, End: 200000, Range: 120000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "foo[2m] @ 180", start: 200000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 60000, End: 180000, Range: 120000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "foo[2m] @ 300", start: 200000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 180000, End: 300000, Range: 120000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "foo[2m] @ 60", start: 200000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: -60000, End: 60000, Range: 120000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "foo[2m] offset 2m", start: 300000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 60000, End: 180000, Range: 120000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "foo[2m] @ 200 offset 2m", start: 300000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: -40000, End: 80000, Range: 120000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "foo[2m:1s]", start: 300000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 175000, End: 300000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "count_over_time(foo[2m:1s])", start: 300000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 175000, End: 300000, Func: "count_over_time"}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "count_over_time(foo[2m:1s] @ 300)", start: 200000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 175000, End: 300000, Func: "count_over_time"}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "count_over_time(foo[2m:1s] @ 200)", start: 200000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 75000, End: 200000, Func: "count_over_time"}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "count_over_time(foo[2m:1s] @ 100)", start: 200000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: -25000, End: 100000, Func: "count_over_time"}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "count_over_time(foo[2m:1s] offset 10s)", start: 300000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							| 
									
										
										
										
											2020-08-11 14:21:39 +08:00
										 |  |  | 			{Start: 165000, End: 290000, Func: "count_over_time"}, | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "count_over_time((foo offset 10s)[2m:1s] offset 10s)", start: 300000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							| 
									
										
										
										
											2020-08-11 14:21:39 +08:00
										 |  |  | 			{Start: 155000, End: 280000, Func: "count_over_time"}, | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 		// When the @ is on the vector selector, the enclosing subquery parameters
 | 
					
						
							|  |  |  | 		// don't affect the hint ranges.
 | 
					
						
							|  |  |  | 		query: "count_over_time((foo @ 200 offset 10s)[2m:1s] offset 10s)", start: 300000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 185000, End: 190000, Func: "count_over_time"}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		// When the @ is on the vector selector, the enclosing subquery parameters
 | 
					
						
							|  |  |  | 		// don't affect the hint ranges.
 | 
					
						
							|  |  |  | 		query: "count_over_time((foo @ 200 offset 10s)[2m:1s] @ 100 offset 10s)", start: 300000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 185000, End: 190000, Func: "count_over_time"}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "count_over_time((foo offset 10s)[2m:1s] @ 100 offset 10s)", start: 300000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: -45000, End: 80000, Func: "count_over_time"}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "foo", start: 10000, end: 20000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 5000, End: 20000, Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "foo @ 15", start: 10000, end: 20000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 10000, End: 15000, Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "foo @ 1", start: 10000, end: 20000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: -4000, End: 1000, Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "rate(foo[2m] @ 180)", start: 200000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 60000, End: 180000, Range: 120000, Func: "rate", Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "rate(foo[2m] @ 300)", start: 200000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 180000, End: 300000, Range: 120000, Func: "rate", Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "rate(foo[2m] @ 60)", start: 200000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: -60000, End: 60000, Range: 120000, Func: "rate", Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "rate(foo[2m])", start: 200000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 80000, End: 500000, Range: 120000, Func: "rate", Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "rate(foo[2m] offset 2m)", start: 300000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 60000, End: 380000, Range: 120000, Func: "rate", Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "rate(foo[2m:1s])", start: 300000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 175000, End: 500000, Func: "rate", Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "count_over_time(foo[2m:1s])", start: 300000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 175000, End: 500000, Func: "count_over_time", Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "count_over_time(foo[2m:1s] offset 10s)", start: 300000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							| 
									
										
										
										
											2020-08-11 14:21:39 +08:00
										 |  |  | 			{Start: 165000, End: 490000, Func: "count_over_time", Step: 1000}, | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "count_over_time(foo[2m:1s] @ 300)", start: 200000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 175000, End: 300000, Func: "count_over_time", Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "count_over_time(foo[2m:1s] @ 200)", start: 200000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 75000, End: 200000, Func: "count_over_time", Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "count_over_time(foo[2m:1s] @ 100)", start: 200000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: -25000, End: 100000, Func: "count_over_time", Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "count_over_time((foo offset 10s)[2m:1s] offset 10s)", start: 300000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							| 
									
										
										
										
											2020-08-11 14:21:39 +08:00
										 |  |  | 			{Start: 155000, End: 480000, Func: "count_over_time", Step: 1000}, | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 	}, { | 
					
						
							|  |  |  | 		// When the @ is on the vector selector, the enclosing subquery parameters
 | 
					
						
							|  |  |  | 		// don't affect the hint ranges.
 | 
					
						
							|  |  |  | 		query: "count_over_time((foo @ 200 offset 10s)[2m:1s] offset 10s)", start: 300000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 185000, End: 190000, Func: "count_over_time", Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		// When the @ is on the vector selector, the enclosing subquery parameters
 | 
					
						
							|  |  |  | 		// don't affect the hint ranges.
 | 
					
						
							|  |  |  | 		query: "count_over_time((foo @ 200 offset 10s)[2m:1s] @ 100 offset 10s)", start: 300000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 185000, End: 190000, Func: "count_over_time", Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "count_over_time((foo offset 10s)[2m:1s] @ 100 offset 10s)", start: 300000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: -45000, End: 80000, Func: "count_over_time", Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-12-05 22:06:28 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "sum by (dim1) (foo)", start: 10000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 5000, End: 10000, Func: "sum", By: true, Grouping: []string{"dim1"}}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-12-05 22:06:28 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "sum without (dim1) (foo)", start: 10000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 5000, End: 10000, Func: "sum", Grouping: []string{"dim1"}}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-12-05 22:06:28 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "sum by (dim1) (avg_over_time(foo[1s]))", start: 10000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 9000, End: 10000, Func: "avg_over_time", Range: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-12-05 22:06:28 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "sum by (dim1) (max by (dim2) (foo))", start: 10000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 5000, End: 10000, Func: "max", By: true, Grouping: []string{"dim2"}}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2019-12-05 22:06:28 +08:00
										 |  |  | 	}, { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		query: "(max by (dim1) (foo))[5s:1s]", start: 10000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 0, End: 10000, Func: "max", By: true, Grouping: []string{"dim1"}}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "(sum(http_requests{group=~\"p.*\"})+max(http_requests{group=~\"c.*\"}))[20s:5s]", start: 120000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 95000, End: 120000, Func: "sum", By: true}, | 
					
						
							|  |  |  | 			{Start: 95000, End: 120000, Func: "max", By: true}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "foo @ 50 + bar @ 250 + baz @ 900", start: 100000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 45000, End: 50000, Step: 1000}, | 
					
						
							|  |  |  | 			{Start: 245000, End: 250000, Step: 1000}, | 
					
						
							|  |  |  | 			{Start: 895000, End: 900000, Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "foo @ 50 + bar + baz @ 900", start: 100000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 45000, End: 50000, Step: 1000}, | 
					
						
							|  |  |  | 			{Start: 95000, End: 500000, Step: 1000}, | 
					
						
							|  |  |  | 			{Start: 895000, End: 900000, Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "rate(foo[2s] @ 50) + bar @ 250 + baz @ 900", start: 100000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 48000, End: 50000, Step: 1000, Func: "rate", Range: 2000}, | 
					
						
							|  |  |  | 			{Start: 245000, End: 250000, Step: 1000}, | 
					
						
							|  |  |  | 			{Start: 895000, End: 900000, Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "rate(foo[2s:1s] @ 50) + bar + baz", start: 100000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 43000, End: 50000, Step: 1000, Func: "rate"}, | 
					
						
							|  |  |  | 			{Start: 95000, End: 500000, Step: 1000}, | 
					
						
							|  |  |  | 			{Start: 95000, End: 500000, Step: 1000}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { | 
					
						
							|  |  |  | 		query: "rate(foo[2s:1s] @ 50) + bar + rate(baz[2m:1s] @ 900 offset 2m) ", start: 100000, end: 500000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 43000, End: 50000, Step: 1000, Func: "rate"}, | 
					
						
							|  |  |  | 			{Start: 95000, End: 500000, Step: 1000}, | 
					
						
							|  |  |  | 			{Start: 655000, End: 780000, Step: 1000, Func: "rate"}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { // Hints are based on the inner most subquery timestamp.
 | 
					
						
							|  |  |  | 		query: `sum_over_time(sum_over_time(metric{job="1"}[100s])[100s:25s] @ 50)[3s:1s] @ 3000`, start: 100000, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: -150000, End: 50000, Range: 100000, Func: "sum_over_time"}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, { // Hints are based on the inner most subquery timestamp.
 | 
					
						
							|  |  |  | 		query: `sum_over_time(sum_over_time(metric{job="1"}[100s])[100s:25s] @ 3000)[3s:1s] @ 50`, | 
					
						
							|  |  |  | 		expected: []*storage.SelectHints{ | 
					
						
							|  |  |  | 			{Start: 2800000, End: 3000000, Range: 100000, Func: "sum_over_time"}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	} { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		t.Run(tc.query, func(t *testing.T) { | 
					
						
							|  |  |  | 			engine := NewEngine(opts) | 
					
						
							|  |  |  | 			hintsRecorder := &noopHintRecordingQueryable{} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			var ( | 
					
						
							|  |  |  | 				query Query | 
					
						
							|  |  |  | 				err   error | 
					
						
							|  |  |  | 			) | 
					
						
							|  |  |  | 			if tc.end == 0 { | 
					
						
							|  |  |  | 				query, err = engine.NewInstantQuery(hintsRecorder, tc.query, timestamp.Time(tc.start)) | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				query, err = engine.NewRangeQuery(hintsRecorder, tc.query, timestamp.Time(tc.start), timestamp.Time(tc.end), time.Second) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 			require.NoError(t, err) | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 			res := query.Exec(context.Background()) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 			require.NoError(t, res.Err) | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 			require.Equal(t, tc.expected, hintsRecorder.hints) | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2019-04-17 20:52:41 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | func TestEngineShutdown(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 	opts := EngineOpts{ | 
					
						
							| 
									
										
										
										
											2020-01-29 04:38:49 +08:00
										 |  |  | 		Logger:     nil, | 
					
						
							|  |  |  | 		Reg:        nil, | 
					
						
							|  |  |  | 		MaxSamples: 10, | 
					
						
							|  |  |  | 		Timeout:    10 * time.Second, | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	engine := NewEngine(opts) | 
					
						
							| 
									
										
										
										
											2016-09-16 06:58:06 +08:00
										 |  |  | 	ctx, cancelCtx := context.WithCancel(context.Background()) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 	block := make(chan struct{}) | 
					
						
							|  |  |  | 	processing := make(chan struct{}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	// Shutdown engine on first handler execution. Should handler execution ever become
 | 
					
						
							|  |  |  | 	// concurrent this test has to be adjusted accordingly.
 | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 	f := func(ctx context.Context) error { | 
					
						
							|  |  |  | 		processing <- struct{}{} | 
					
						
							|  |  |  | 		<-block | 
					
						
							|  |  |  | 		return contextDone(ctx, "test statement execution") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	query1 := engine.newTestQuery(f) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-29 17:08:56 +08:00
										 |  |  | 	// Stopping the engine must cancel the base context. While executing queries is
 | 
					
						
							|  |  |  | 	// still possible, their context is canceled from the beginning and execution should
 | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 	// terminate immediately.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 	var res *Result | 
					
						
							|  |  |  | 	go func() { | 
					
						
							| 
									
										
											  
											
												promql: Allow per-query contexts.
For Weaveworks' Frankenstein, we need to support multitenancy. In
Frankenstein, we initially solved this without modifying the promql
package at all: we constructed a new promql.Engine for every
query and injected a storage implementation into that engine which would
be primed to only collect data for a given user.
This is problematic to upstream, however. Prometheus assumes that there
is only one engine: the query concurrency gate is part of the engine,
and the engine contains one central cancellable context to shut down all
queries. Also, creating a new engine for every query seems like overkill.
Thus, we want to be able to pass per-query contexts into a single engine.
This change gets rid of the promql.Engine's built-in base context and
allows passing in a per-query context instead. Central cancellation of
all queries is still possible by deriving all passed-in contexts from
one central one, but this is now the responsibility of the caller. The
central query context is now created in main() and passed into the
relevant components (web handler / API, rule manager).
In a next step, the per-query context would have to be passed to the
storage implementation, so that the storage can implement multi-tenancy
or other features based on the contextual information.
											
										 
											2016-09-15 19:52:50 +08:00
										 |  |  | 		res = query1.Exec(ctx) | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 		processing <- struct{}{} | 
					
						
							|  |  |  | 	}() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	<-processing | 
					
						
							| 
									
										
										
										
											2016-09-16 06:58:06 +08:00
										 |  |  | 	cancelCtx() | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 	block <- struct{}{} | 
					
						
							|  |  |  | 	<-processing | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.Error(t, res.Err, "expected error on shutdown during query but got none") | 
					
						
							|  |  |  | 	require.Equal(t, errQueryCanceled, res.Err) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-10 20:21:24 +08:00
										 |  |  | 	query2 := engine.newTestQuery(func(context.Context) error { | 
					
						
							|  |  |  | 		t.Fatalf("reached query execution unexpectedly") | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// The second query is started after the engine shut down. It must
 | 
					
						
							|  |  |  | 	// be canceled immediately.
 | 
					
						
							| 
									
										
											  
											
												promql: Allow per-query contexts.
For Weaveworks' Frankenstein, we need to support multitenancy. In
Frankenstein, we initially solved this without modifying the promql
package at all: we constructed a new promql.Engine for every
query and injected a storage implementation into that engine which would
be primed to only collect data for a given user.
This is problematic to upstream, however. Prometheus assumes that there
is only one engine: the query concurrency gate is part of the engine,
and the engine contains one central cancellable context to shut down all
queries. Also, creating a new engine for every query seems like overkill.
Thus, we want to be able to pass per-query contexts into a single engine.
This change gets rid of the promql.Engine's built-in base context and
allows passing in a per-query context instead. Central cancellation of
all queries is still possible by deriving all passed-in contexts from
one central one, but this is now the responsibility of the caller. The
central query context is now created in main() and passed into the
relevant components (web handler / API, rule manager).
In a next step, the per-query context would have to be passed to the
storage implementation, so that the storage can implement multi-tenancy
or other features based on the contextual information.
											
										 
											2016-09-15 19:52:50 +08:00
										 |  |  | 	res2 := query2.Exec(ctx) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.Error(t, res2.Err, "expected error on querying with canceled context but got none") | 
					
						
							| 
									
										
										
										
											2019-10-10 08:06:53 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	var e ErrQueryCanceled | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.True(t, errors.As(res2.Err, &e), "expected cancellation error but got: %s", res2.Err) | 
					
						
							| 
									
										
										
										
											2015-03-31 01:13:36 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-08-19 21:28:53 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-18 21:51:10 +08:00
										 |  |  | func TestEngineEvalStmtTimestamps(t *testing.T) { | 
					
						
							|  |  |  | 	test, err := NewTest(t, ` | 
					
						
							|  |  |  | load 10s | 
					
						
							|  |  |  |   metric 1 2 | 
					
						
							|  |  |  | `) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.NoError(t, err) | 
					
						
							| 
									
										
										
										
											2018-03-09 00:48:11 +08:00
										 |  |  | 	defer test.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-18 21:51:10 +08:00
										 |  |  | 	err = test.Run() | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.NoError(t, err) | 
					
						
							| 
									
										
										
										
											2017-04-18 21:51:10 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	cases := []struct { | 
					
						
							| 
									
										
										
										
											2018-09-13 17:57:36 +08:00
										 |  |  | 		Query       string | 
					
						
							| 
									
										
										
										
											2020-02-04 00:32:23 +08:00
										 |  |  | 		Result      parser.Value | 
					
						
							| 
									
										
										
										
											2018-09-13 17:57:36 +08:00
										 |  |  | 		Start       time.Time | 
					
						
							|  |  |  | 		End         time.Time | 
					
						
							|  |  |  | 		Interval    time.Duration | 
					
						
							|  |  |  | 		ShouldError bool | 
					
						
							| 
									
										
										
										
											2017-04-18 21:51:10 +08:00
										 |  |  | 	}{ | 
					
						
							|  |  |  | 		// Instant queries.
 | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Query:  "1", | 
					
						
							|  |  |  | 			Result: Scalar{V: 1, T: 1000}, | 
					
						
							|  |  |  | 			Start:  time.Unix(1, 0), | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Query: "metric", | 
					
						
							|  |  |  | 			Result: Vector{ | 
					
						
							|  |  |  | 				Sample{Point: Point{V: 1, T: 1000}, | 
					
						
							|  |  |  | 					Metric: labels.FromStrings("__name__", "metric")}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			Start: time.Unix(1, 0), | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Query: "metric[20s]", | 
					
						
							|  |  |  | 			Result: Matrix{Series{ | 
					
						
							|  |  |  | 				Points: []Point{{V: 1, T: 0}, {V: 2, T: 10000}}, | 
					
						
							|  |  |  | 				Metric: labels.FromStrings("__name__", "metric")}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			Start: time.Unix(10, 0), | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		// Range queries.
 | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Query: "1", | 
					
						
							|  |  |  | 			Result: Matrix{Series{ | 
					
						
							|  |  |  | 				Points: []Point{{V: 1, T: 0}, {V: 1, T: 1000}, {V: 1, T: 2000}}, | 
					
						
							|  |  |  | 				Metric: labels.FromStrings()}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			Start:    time.Unix(0, 0), | 
					
						
							|  |  |  | 			End:      time.Unix(2, 0), | 
					
						
							|  |  |  | 			Interval: time.Second, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Query: "metric", | 
					
						
							|  |  |  | 			Result: Matrix{Series{ | 
					
						
							| 
									
										
										
										
											2017-06-13 13:22:27 +08:00
										 |  |  | 				Points: []Point{{V: 1, T: 0}, {V: 1, T: 1000}, {V: 1, T: 2000}}, | 
					
						
							| 
									
										
										
										
											2017-04-18 21:51:10 +08:00
										 |  |  | 				Metric: labels.FromStrings("__name__", "metric")}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			Start:    time.Unix(0, 0), | 
					
						
							|  |  |  | 			End:      time.Unix(2, 0), | 
					
						
							|  |  |  | 			Interval: time.Second, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2017-06-13 13:22:27 +08:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 			Query: "metric", | 
					
						
							|  |  |  | 			Result: Matrix{Series{ | 
					
						
							|  |  |  | 				Points: []Point{{V: 1, T: 0}, {V: 1, T: 5000}, {V: 2, T: 10000}}, | 
					
						
							|  |  |  | 				Metric: labels.FromStrings("__name__", "metric")}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 			Start:    time.Unix(0, 0), | 
					
						
							|  |  |  | 			End:      time.Unix(10, 0), | 
					
						
							|  |  |  | 			Interval: 5 * time.Second, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2018-09-13 17:57:36 +08:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 			Query:       `count_values("wrong label!", metric)`, | 
					
						
							|  |  |  | 			ShouldError: true, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2017-04-18 21:51:10 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, c := range cases { | 
					
						
							|  |  |  | 		var err error | 
					
						
							|  |  |  | 		var qry Query | 
					
						
							|  |  |  | 		if c.Interval == 0 { | 
					
						
							| 
									
										
										
										
											2018-01-10 00:44:23 +08:00
										 |  |  | 			qry, err = test.QueryEngine().NewInstantQuery(test.Queryable(), c.Query, c.Start) | 
					
						
							| 
									
										
										
										
											2017-04-18 21:51:10 +08:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2018-01-10 00:44:23 +08:00
										 |  |  | 			qry, err = test.QueryEngine().NewRangeQuery(test.Queryable(), c.Query, c.Start, c.End, c.Interval) | 
					
						
							| 
									
										
										
										
											2017-04-18 21:51:10 +08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 		require.NoError(t, err) | 
					
						
							| 
									
										
										
										
											2019-10-10 08:06:53 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-18 21:51:10 +08:00
										 |  |  | 		res := qry.Exec(test.Context()) | 
					
						
							| 
									
										
										
										
											2018-09-13 17:57:36 +08:00
										 |  |  | 		if c.ShouldError { | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 			require.Error(t, res.Err, "expected error for the query %q", c.Query) | 
					
						
							| 
									
										
										
										
											2018-09-13 17:57:36 +08:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-10-10 08:06:53 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 		require.NoError(t, res.Err) | 
					
						
							|  |  |  | 		require.Equal(t, c.Result, res.Value, "query %q failed", c.Query) | 
					
						
							| 
									
										
										
										
											2017-04-18 21:51:10 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | func TestMaxQuerySamples(t *testing.T) { | 
					
						
							|  |  |  | 	test, err := NewTest(t, ` | 
					
						
							|  |  |  | load 10s | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  |   metric 1+1x100 | 
					
						
							|  |  |  |   bigmetric{a="1"} 1+1x100 | 
					
						
							|  |  |  |   bigmetric{a="2"} 1+1x100 | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | `) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.NoError(t, err) | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 	defer test.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err = test.Run() | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.NoError(t, err) | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 	// These test cases should be touching the limit exactly (hence no exceeding).
 | 
					
						
							|  |  |  | 	// Exceeding the limit will be tested by doing -1 to the MaxSamples.
 | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 	cases := []struct { | 
					
						
							|  |  |  | 		Query      string | 
					
						
							|  |  |  | 		MaxSamples int | 
					
						
							|  |  |  | 		Start      time.Time | 
					
						
							|  |  |  | 		End        time.Time | 
					
						
							|  |  |  | 		Interval   time.Duration | 
					
						
							|  |  |  | 	}{ | 
					
						
							|  |  |  | 		// Instant queries.
 | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Query:      "1", | 
					
						
							|  |  |  | 			MaxSamples: 1, | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 			Start:      time.Unix(1, 0), | 
					
						
							|  |  |  | 		}, { | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 			Query:      "metric", | 
					
						
							|  |  |  | 			MaxSamples: 1, | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 			Start:      time.Unix(1, 0), | 
					
						
							|  |  |  | 		}, { | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 			Query:      "metric[20s]", | 
					
						
							|  |  |  | 			MaxSamples: 2, | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 			Start:      time.Unix(10, 0), | 
					
						
							|  |  |  | 		}, { | 
					
						
							| 
									
										
										
										
											2018-12-22 21:47:13 +08:00
										 |  |  | 			Query:      "rate(metric[20s])", | 
					
						
							|  |  |  | 			MaxSamples: 3, | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 			Start:      time.Unix(10, 0), | 
					
						
							|  |  |  | 		}, { | 
					
						
							| 
									
										
										
										
											2018-12-22 21:47:13 +08:00
										 |  |  | 			Query:      "metric[20s:5s]", | 
					
						
							|  |  |  | 			MaxSamples: 3, | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 			Start:      time.Unix(10, 0), | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 		}, { | 
					
						
							|  |  |  | 			Query:      "metric[20s] @ 10", | 
					
						
							|  |  |  | 			MaxSamples: 2, | 
					
						
							|  |  |  | 			Start:      time.Unix(0, 0), | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 		}, | 
					
						
							|  |  |  | 		// Range queries.
 | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Query:      "1", | 
					
						
							|  |  |  | 			MaxSamples: 3, | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 			Start:      time.Unix(0, 0), | 
					
						
							|  |  |  | 			End:        time.Unix(2, 0), | 
					
						
							|  |  |  | 			Interval:   time.Second, | 
					
						
							|  |  |  | 		}, { | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 			Query:      "1", | 
					
						
							|  |  |  | 			MaxSamples: 3, | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 			Start:      time.Unix(0, 0), | 
					
						
							|  |  |  | 			End:        time.Unix(2, 0), | 
					
						
							|  |  |  | 			Interval:   time.Second, | 
					
						
							|  |  |  | 		}, { | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 			Query:      "metric", | 
					
						
							|  |  |  | 			MaxSamples: 3, | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 			Start:      time.Unix(0, 0), | 
					
						
							|  |  |  | 			End:        time.Unix(2, 0), | 
					
						
							|  |  |  | 			Interval:   time.Second, | 
					
						
							|  |  |  | 		}, { | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 			Query:      "metric", | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 			MaxSamples: 3, | 
					
						
							|  |  |  | 			Start:      time.Unix(0, 0), | 
					
						
							|  |  |  | 			End:        time.Unix(10, 0), | 
					
						
							|  |  |  | 			Interval:   5 * time.Second, | 
					
						
							|  |  |  | 		}, { | 
					
						
							| 
									
										
										
										
											2020-05-28 20:36:30 +08:00
										 |  |  | 			Query:      "rate(bigmetric[1s])", | 
					
						
							|  |  |  | 			MaxSamples: 1, | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 			Start:      time.Unix(0, 0), | 
					
						
							|  |  |  | 			End:        time.Unix(10, 0), | 
					
						
							|  |  |  | 			Interval:   5 * time.Second, | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 		}, { | 
					
						
							|  |  |  | 			// Result is duplicated, so @ also produces 3 samples.
 | 
					
						
							|  |  |  | 			Query:      "metric @ 10", | 
					
						
							|  |  |  | 			MaxSamples: 3, | 
					
						
							|  |  |  | 			Start:      time.Unix(0, 0), | 
					
						
							|  |  |  | 			End:        time.Unix(10, 0), | 
					
						
							|  |  |  | 			Interval:   5 * time.Second, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			// The peak samples in memory is during the first evaluation:
 | 
					
						
							|  |  |  | 			//   - Subquery takes 22 samples, 11 for each bigmetric,
 | 
					
						
							|  |  |  | 			//   - Result is calculated per series where the series samples is buffered, hence 11 more here.
 | 
					
						
							|  |  |  | 			//   - The result of two series is added before the last series buffer is discarded, so 2 more here.
 | 
					
						
							|  |  |  | 			//   Hence at peak it is 22 (subquery) + 11 (buffer of a series) + 2 (result from 2 series).
 | 
					
						
							|  |  |  | 			// The subquery samples and the buffer is discarded before duplicating.
 | 
					
						
							|  |  |  | 			Query:      `rate(bigmetric[10s:1s] @ 10)`, | 
					
						
							|  |  |  | 			MaxSamples: 35, | 
					
						
							|  |  |  | 			Start:      time.Unix(0, 0), | 
					
						
							|  |  |  | 			End:        time.Unix(10, 0), | 
					
						
							|  |  |  | 			Interval:   5 * time.Second, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			// Here the reasoning is same as above. But LHS and RHS are done one after another.
 | 
					
						
							|  |  |  | 			// So while one of them takes 35 samples at peak, we need to hold the 2 sample
 | 
					
						
							|  |  |  | 			// result of the other till then.
 | 
					
						
							|  |  |  | 			Query:      `rate(bigmetric[10s:1s] @ 10) + rate(bigmetric[10s:1s] @ 30)`, | 
					
						
							|  |  |  | 			MaxSamples: 37, | 
					
						
							|  |  |  | 			Start:      time.Unix(0, 0), | 
					
						
							|  |  |  | 			End:        time.Unix(10, 0), | 
					
						
							|  |  |  | 			Interval:   5 * time.Second, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			// Sample as above but with only 1 part as step invariant.
 | 
					
						
							|  |  |  | 			// Here the peak is caused by the non-step invariant part as it touches more time range.
 | 
					
						
							|  |  |  | 			// Hence at peak it is 2*21 (subquery from 0s to 20s)
 | 
					
						
							|  |  |  | 			//                     + 11 (buffer of a series per evaluation)
 | 
					
						
							|  |  |  | 			//                     + 6 (result from 2 series at 3 eval times).
 | 
					
						
							|  |  |  | 			Query:      `rate(bigmetric[10s:1s]) + rate(bigmetric[10s:1s] @ 30)`, | 
					
						
							|  |  |  | 			MaxSamples: 59, | 
					
						
							|  |  |  | 			Start:      time.Unix(10, 0), | 
					
						
							|  |  |  | 			End:        time.Unix(20, 0), | 
					
						
							|  |  |  | 			Interval:   5 * time.Second, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			// Nested subquery.
 | 
					
						
							|  |  |  | 			// We saw that innermost rate takes 35 samples which is still the peak
 | 
					
						
							|  |  |  | 			// since the other two subqueries just duplicate the result.
 | 
					
						
							|  |  |  | 			Query:      `rate(rate(bigmetric[10s:1s] @ 10)[100s:25s] @ 1000)[100s:20s] @ 2000`, | 
					
						
							|  |  |  | 			MaxSamples: 35, | 
					
						
							|  |  |  | 			Start:      time.Unix(10, 0), | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			// Nested subquery.
 | 
					
						
							|  |  |  | 			// Now the outmost subquery produces more samples than inner most rate.
 | 
					
						
							|  |  |  | 			Query:      `rate(rate(bigmetric[10s:1s] @ 10)[100s:25s] @ 1000)[17s:1s] @ 2000`, | 
					
						
							|  |  |  | 			MaxSamples: 36, | 
					
						
							|  |  |  | 			Start:      time.Unix(10, 0), | 
					
						
							| 
									
										
										
										
											2020-05-28 20:36:30 +08:00
										 |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	engine := test.QueryEngine() | 
					
						
							|  |  |  | 	for _, c := range cases { | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 		t.Run(c.Query, func(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 			testFunc := func(expError error) { | 
					
						
							|  |  |  | 				var err error | 
					
						
							|  |  |  | 				var qry Query | 
					
						
							|  |  |  | 				if c.Interval == 0 { | 
					
						
							|  |  |  | 					qry, err = engine.NewInstantQuery(test.Queryable(), c.Query, c.Start) | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					qry, err = engine.NewRangeQuery(test.Queryable(), c.Query, c.Start, c.End, c.Interval) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				require.NoError(t, err) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				res := qry.Exec(test.Context()) | 
					
						
							|  |  |  | 				require.Equal(t, expError, res.Err) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 			// Within limit.
 | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 			engine.maxSamplesPerQuery = c.MaxSamples | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 			testFunc(nil) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// Exceeding limit.
 | 
					
						
							|  |  |  | 			engine.maxSamplesPerQuery = c.MaxSamples - 1 | 
					
						
							|  |  |  | 			testFunc(ErrTooManySamples(env)) | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestAtModifier(t *testing.T) { | 
					
						
							|  |  |  | 	test, err := NewTest(t, ` | 
					
						
							|  |  |  | load 10s | 
					
						
							|  |  |  |   metric{job="1"} 0+1x1000 | 
					
						
							|  |  |  |   metric{job="2"} 0+2x1000 | 
					
						
							|  |  |  |   metric_topk{instance="1"} 0+1x1000 | 
					
						
							|  |  |  |   metric_topk{instance="2"} 0+2x1000 | 
					
						
							|  |  |  |   metric_topk{instance="3"} 1000-1x1000 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | load 1ms | 
					
						
							|  |  |  |   metric_ms 0+1x10000 | 
					
						
							|  |  |  | `) | 
					
						
							|  |  |  | 	require.NoError(t, err) | 
					
						
							|  |  |  | 	defer test.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err = test.Run() | 
					
						
							|  |  |  | 	require.NoError(t, err) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	lbls1 := labels.FromStrings("__name__", "metric", "job", "1") | 
					
						
							|  |  |  | 	lbls2 := labels.FromStrings("__name__", "metric", "job", "2") | 
					
						
							|  |  |  | 	lblstopk2 := labels.FromStrings("__name__", "metric_topk", "instance", "2") | 
					
						
							|  |  |  | 	lblstopk3 := labels.FromStrings("__name__", "metric_topk", "instance", "3") | 
					
						
							|  |  |  | 	lblsms := labels.FromStrings("__name__", "metric_ms") | 
					
						
							|  |  |  | 	lblsneg := labels.FromStrings("__name__", "metric_neg") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Add some samples with negative timestamp.
 | 
					
						
							|  |  |  | 	db := test.TSDB() | 
					
						
							|  |  |  | 	app := db.Appender(context.Background()) | 
					
						
							| 
									
										
										
										
											2021-02-18 20:07:00 +08:00
										 |  |  | 	ref, err := app.Append(0, lblsneg, -1000000, 1000) | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 	require.NoError(t, err) | 
					
						
							|  |  |  | 	for ts := int64(-1000000 + 1000); ts <= 0; ts += 1000 { | 
					
						
							| 
									
										
										
										
											2021-02-18 20:07:00 +08:00
										 |  |  | 		_, err := app.Append(ref, nil, ts, -float64(ts/1000)+1) | 
					
						
							|  |  |  | 		require.NoError(t, err) | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-02-03 21:43:12 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// To test the fix for https://github.com/prometheus/prometheus/issues/8433.
 | 
					
						
							| 
									
										
										
										
											2021-02-18 20:07:00 +08:00
										 |  |  | 	_, err = app.Append(0, labels.FromStrings("__name__", "metric_timestamp"), 3600*1000, 1000) | 
					
						
							| 
									
										
										
										
											2021-02-03 21:43:12 +08:00
										 |  |  | 	require.NoError(t, err) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 	require.NoError(t, app.Commit()) | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 	cases := []struct { | 
					
						
							|  |  |  | 		query                string | 
					
						
							|  |  |  | 		start, end, interval int64 // Time in seconds.
 | 
					
						
							|  |  |  | 		result               parser.Value | 
					
						
							|  |  |  | 	}{ | 
					
						
							|  |  |  | 		{ // Time of the result is the evaluation time.
 | 
					
						
							|  |  |  | 			query: `metric_neg @ 0`, | 
					
						
							|  |  |  | 			start: 100, | 
					
						
							|  |  |  | 			result: Vector{ | 
					
						
							|  |  |  | 				Sample{Point: Point{V: 1, T: 100000}, Metric: lblsneg}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			query: `metric_neg @ -200`, | 
					
						
							|  |  |  | 			start: 100, | 
					
						
							|  |  |  | 			result: Vector{ | 
					
						
							|  |  |  | 				Sample{Point: Point{V: 201, T: 100000}, Metric: lblsneg}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			query: `metric{job="2"} @ 50`, | 
					
						
							|  |  |  | 			start: -2, end: 2, interval: 1, | 
					
						
							|  |  |  | 			result: Matrix{ | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{{V: 10, T: -2000}, {V: 10, T: -1000}, {V: 10, T: 0}, {V: 10, T: 1000}, {V: 10, T: 2000}}, | 
					
						
							|  |  |  | 					Metric: lbls2, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { // Timestamps for matrix selector does not depend on the evaluation time.
 | 
					
						
							|  |  |  | 			query: "metric[20s] @ 300", | 
					
						
							|  |  |  | 			start: 10, | 
					
						
							|  |  |  | 			result: Matrix{ | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{{V: 28, T: 280000}, {V: 29, T: 290000}, {V: 30, T: 300000}}, | 
					
						
							|  |  |  | 					Metric: lbls1, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{{V: 56, T: 280000}, {V: 58, T: 290000}, {V: 60, T: 300000}}, | 
					
						
							|  |  |  | 					Metric: lbls2, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			query: `metric_neg[2s] @ 0`, | 
					
						
							|  |  |  | 			start: 100, | 
					
						
							|  |  |  | 			result: Matrix{ | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{{V: 3, T: -2000}, {V: 2, T: -1000}, {V: 1, T: 0}}, | 
					
						
							|  |  |  | 					Metric: lblsneg, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			query: `metric_neg[3s] @ -500`, | 
					
						
							|  |  |  | 			start: 100, | 
					
						
							|  |  |  | 			result: Matrix{ | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{{V: 504, T: -503000}, {V: 503, T: -502000}, {V: 502, T: -501000}, {V: 501, T: -500000}}, | 
					
						
							|  |  |  | 					Metric: lblsneg, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			query: `metric_ms[3ms] @ 2.345`, | 
					
						
							|  |  |  | 			start: 100, | 
					
						
							|  |  |  | 			result: Matrix{ | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{{V: 2342, T: 2342}, {V: 2343, T: 2343}, {V: 2344, T: 2344}, {V: 2345, T: 2345}}, | 
					
						
							|  |  |  | 					Metric: lblsms, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			query: "metric[100s:25s] @ 300", | 
					
						
							|  |  |  | 			start: 100, | 
					
						
							|  |  |  | 			result: Matrix{ | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{{V: 20, T: 200000}, {V: 22, T: 225000}, {V: 25, T: 250000}, {V: 27, T: 275000}, {V: 30, T: 300000}}, | 
					
						
							|  |  |  | 					Metric: lbls1, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{{V: 40, T: 200000}, {V: 44, T: 225000}, {V: 50, T: 250000}, {V: 54, T: 275000}, {V: 60, T: 300000}}, | 
					
						
							|  |  |  | 					Metric: lbls2, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			query: "metric_neg[50s:25s] @ 0", | 
					
						
							|  |  |  | 			start: 100, | 
					
						
							|  |  |  | 			result: Matrix{ | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{{V: 51, T: -50000}, {V: 26, T: -25000}, {V: 1, T: 0}}, | 
					
						
							|  |  |  | 					Metric: lblsneg, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			query: "metric_neg[50s:25s] @ -100", | 
					
						
							|  |  |  | 			start: 100, | 
					
						
							|  |  |  | 			result: Matrix{ | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{{V: 151, T: -150000}, {V: 126, T: -125000}, {V: 101, T: -100000}}, | 
					
						
							|  |  |  | 					Metric: lblsneg, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			query: `metric_ms[100ms:25ms] @ 2.345`, | 
					
						
							|  |  |  | 			start: 100, | 
					
						
							|  |  |  | 			result: Matrix{ | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{{V: 2250, T: 2250}, {V: 2275, T: 2275}, {V: 2300, T: 2300}, {V: 2325, T: 2325}}, | 
					
						
							|  |  |  | 					Metric: lblsms, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			query: `metric_topk and topk(1, sum_over_time(metric_topk[50s] @ 100))`, | 
					
						
							|  |  |  | 			start: 50, end: 80, interval: 10, | 
					
						
							|  |  |  | 			result: Matrix{ | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{{V: 995, T: 50000}, {V: 994, T: 60000}, {V: 993, T: 70000}, {V: 992, T: 80000}}, | 
					
						
							|  |  |  | 					Metric: lblstopk3, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			query: `metric_topk and topk(1, sum_over_time(metric_topk[50s] @ 5000))`, | 
					
						
							|  |  |  | 			start: 50, end: 80, interval: 10, | 
					
						
							|  |  |  | 			result: Matrix{ | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{{V: 10, T: 50000}, {V: 12, T: 60000}, {V: 14, T: 70000}, {V: 16, T: 80000}}, | 
					
						
							|  |  |  | 					Metric: lblstopk2, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							| 
									
										
										
										
											2021-02-10 00:03:16 +08:00
										 |  |  | 		}, { | 
					
						
							|  |  |  | 			query: `metric_topk and topk(1, sum_over_time(metric_topk[50s] @ end()))`, | 
					
						
							|  |  |  | 			start: 70, end: 100, interval: 10, | 
					
						
							|  |  |  | 			result: Matrix{ | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{{V: 993, T: 70000}, {V: 992, T: 80000}, {V: 991, T: 90000}, {V: 990, T: 100000}}, | 
					
						
							|  |  |  | 					Metric: lblstopk3, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			query: `metric_topk and topk(1, sum_over_time(metric_topk[50s] @ start()))`, | 
					
						
							|  |  |  | 			start: 100, end: 130, interval: 10, | 
					
						
							|  |  |  | 			result: Matrix{ | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{{V: 990, T: 100000}, {V: 989, T: 110000}, {V: 988, T: 120000}, {V: 987, T: 130000}}, | 
					
						
							|  |  |  | 					Metric: lblstopk3, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							| 
									
										
										
										
											2021-02-03 21:43:12 +08:00
										 |  |  | 		}, { | 
					
						
							|  |  |  | 			// Tests for https://github.com/prometheus/prometheus/issues/8433.
 | 
					
						
							|  |  |  | 			// The trick here is that the query range should be > lookback delta.
 | 
					
						
							|  |  |  | 			query: `timestamp(metric_timestamp @ 3600)`, | 
					
						
							|  |  |  | 			start: 0, end: 7 * 60, interval: 60, | 
					
						
							|  |  |  | 			result: Matrix{ | 
					
						
							|  |  |  | 				Series{ | 
					
						
							|  |  |  | 					Points: []Point{ | 
					
						
							|  |  |  | 						{V: 3600, T: 0}, | 
					
						
							|  |  |  | 						{V: 3600, T: 60 * 1000}, | 
					
						
							|  |  |  | 						{V: 3600, T: 2 * 60 * 1000}, | 
					
						
							|  |  |  | 						{V: 3600, T: 3 * 60 * 1000}, | 
					
						
							|  |  |  | 						{V: 3600, T: 4 * 60 * 1000}, | 
					
						
							|  |  |  | 						{V: 3600, T: 5 * 60 * 1000}, | 
					
						
							|  |  |  | 						{V: 3600, T: 6 * 60 * 1000}, | 
					
						
							|  |  |  | 						{V: 3600, T: 7 * 60 * 1000}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Metric: labels.Labels{}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, c := range cases { | 
					
						
							|  |  |  | 		t.Run(c.query, func(t *testing.T) { | 
					
						
							|  |  |  | 			if c.interval == 0 { | 
					
						
							|  |  |  | 				c.interval = 1 | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			start, end, interval := time.Unix(c.start, 0), time.Unix(c.end, 0), time.Duration(c.interval)*time.Second | 
					
						
							|  |  |  | 			var err error | 
					
						
							|  |  |  | 			var qry Query | 
					
						
							|  |  |  | 			if c.end == 0 { | 
					
						
							|  |  |  | 				qry, err = test.QueryEngine().NewInstantQuery(test.Queryable(), c.query, start) | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 				qry, err = test.QueryEngine().NewRangeQuery(test.Queryable(), c.query, start, end, interval) | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			require.NoError(t, err) | 
					
						
							| 
									
										
										
										
											2019-10-10 08:06:53 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 			res := qry.Exec(test.Context()) | 
					
						
							|  |  |  | 			require.NoError(t, res.Err) | 
					
						
							| 
									
										
										
										
											2021-01-24 22:53:30 +08:00
										 |  |  | 			if expMat, ok := c.result.(Matrix); ok { | 
					
						
							|  |  |  | 				sort.Sort(expMat) | 
					
						
							|  |  |  | 				sort.Sort(res.Value.(Matrix)) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 			require.Equal(t, c.result, res.Value, "query %q failed", c.query) | 
					
						
							| 
									
										
										
										
											2021-01-04 17:03:36 +08:00
										 |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2018-10-02 19:59:19 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-02-03 21:43:12 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-19 21:28:53 +08:00
										 |  |  | func TestRecoverEvaluatorRuntime(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2017-08-12 02:45:52 +08:00
										 |  |  | 	ev := &evaluator{logger: log.NewNopLogger()} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-19 21:28:53 +08:00
										 |  |  | 	var err error | 
					
						
							| 
									
										
										
										
											2020-06-10 00:57:31 +08:00
										 |  |  | 	defer ev.recover(nil, &err) | 
					
						
							| 
									
										
										
										
											2015-08-19 21:28:53 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Cause a runtime panic.
 | 
					
						
							|  |  |  | 	var a []int | 
					
						
							| 
									
										
										
										
											2019-05-03 21:11:28 +08:00
										 |  |  | 	//nolint:govet
 | 
					
						
							| 
									
										
										
										
											2015-08-19 21:28:53 +08:00
										 |  |  | 	a[123] = 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if err.Error() != "unexpected error" { | 
					
						
							|  |  |  | 		t.Fatalf("wrong error message: %q, expected %q", err, "unexpected error") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestRecoverEvaluatorError(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2017-08-12 02:45:52 +08:00
										 |  |  | 	ev := &evaluator{logger: log.NewNopLogger()} | 
					
						
							| 
									
										
										
										
											2015-08-19 21:28:53 +08:00
										 |  |  | 	var err error | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-26 07:01:12 +08:00
										 |  |  | 	e := errors.New("custom error") | 
					
						
							| 
									
										
										
										
											2015-08-19 21:28:53 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-26 08:04:01 +08:00
										 |  |  | 	defer func() { | 
					
						
							|  |  |  | 		if err.Error() != e.Error() { | 
					
						
							|  |  |  | 			t.Fatalf("wrong error message: %q, expected %q", err, e) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}() | 
					
						
							| 
									
										
										
										
											2020-06-10 00:57:31 +08:00
										 |  |  | 	defer ev.recover(nil, &err) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	panic(e) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestRecoverEvaluatorErrorWithWarnings(t *testing.T) { | 
					
						
							|  |  |  | 	ev := &evaluator{logger: log.NewNopLogger()} | 
					
						
							|  |  |  | 	var err error | 
					
						
							|  |  |  | 	var ws storage.Warnings | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	warnings := storage.Warnings{errors.New("custom warning")} | 
					
						
							|  |  |  | 	e := errWithWarnings{ | 
					
						
							|  |  |  | 		err:      errors.New("custom error"), | 
					
						
							|  |  |  | 		warnings: warnings, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	defer func() { | 
					
						
							|  |  |  | 		if err.Error() != e.Error() { | 
					
						
							|  |  |  | 			t.Fatalf("wrong error message: %q, expected %q", err, e) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if len(ws) != len(warnings) && ws[0] != warnings[0] { | 
					
						
							|  |  |  | 			t.Fatalf("wrong warning message: %q, expected %q", ws[0], warnings[0]) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}() | 
					
						
							|  |  |  | 	defer ev.recover(&ws, &err) | 
					
						
							| 
									
										
										
										
											2015-08-26 08:04:01 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	panic(e) | 
					
						
							| 
									
										
										
										
											2015-08-19 21:28:53 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-12-22 21:47:13 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | func TestSubquerySelector(t *testing.T) { | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 	type caseType struct { | 
					
						
							|  |  |  | 		Query  string | 
					
						
							|  |  |  | 		Result Result | 
					
						
							|  |  |  | 		Start  time.Time | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, tst := range []struct { | 
					
						
							| 
									
										
										
										
											2018-12-22 21:47:13 +08:00
										 |  |  | 		loadString string | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 		cases      []caseType | 
					
						
							| 
									
										
										
										
											2018-12-22 21:47:13 +08:00
										 |  |  | 	}{ | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			loadString: `load 10s | 
					
						
							|  |  |  | 							metric 1 2`, | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 			cases: []caseType{ | 
					
						
							| 
									
										
										
										
											2018-12-22 21:47:13 +08:00
										 |  |  | 				{ | 
					
						
							|  |  |  | 					Query: "metric[20s:10s]", | 
					
						
							|  |  |  | 					Result: Result{ | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 						Matrix{Series{ | 
					
						
							|  |  |  | 							Points: []Point{{V: 1, T: 0}, {V: 2, T: 10000}}, | 
					
						
							|  |  |  | 							Metric: labels.FromStrings("__name__", "metric")}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Start: time.Unix(10, 0), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					Query: "metric[20s:5s]", | 
					
						
							|  |  |  | 					Result: Result{ | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 						Matrix{Series{ | 
					
						
							|  |  |  | 							Points: []Point{{V: 1, T: 0}, {V: 1, T: 5000}, {V: 2, T: 10000}}, | 
					
						
							|  |  |  | 							Metric: labels.FromStrings("__name__", "metric")}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Start: time.Unix(10, 0), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					Query: "metric[20s:5s] offset 2s", | 
					
						
							|  |  |  | 					Result: Result{ | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 						Matrix{Series{ | 
					
						
							|  |  |  | 							Points: []Point{{V: 1, T: 0}, {V: 1, T: 5000}, {V: 2, T: 10000}}, | 
					
						
							|  |  |  | 							Metric: labels.FromStrings("__name__", "metric")}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Start: time.Unix(12, 0), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					Query: "metric[20s:5s] offset 6s", | 
					
						
							|  |  |  | 					Result: Result{ | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 						Matrix{Series{ | 
					
						
							|  |  |  | 							Points: []Point{{V: 1, T: 0}, {V: 1, T: 5000}, {V: 2, T: 10000}}, | 
					
						
							|  |  |  | 							Metric: labels.FromStrings("__name__", "metric")}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Start: time.Unix(20, 0), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					Query: "metric[20s:5s] offset 4s", | 
					
						
							|  |  |  | 					Result: Result{ | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 						Matrix{Series{ | 
					
						
							|  |  |  | 							Points: []Point{{V: 2, T: 15000}, {V: 2, T: 20000}, {V: 2, T: 25000}, {V: 2, T: 30000}}, | 
					
						
							|  |  |  | 							Metric: labels.FromStrings("__name__", "metric")}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Start: time.Unix(35, 0), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					Query: "metric[20s:5s] offset 5s", | 
					
						
							|  |  |  | 					Result: Result{ | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 						Matrix{Series{ | 
					
						
							|  |  |  | 							Points: []Point{{V: 2, T: 10000}, {V: 2, T: 15000}, {V: 2, T: 20000}, {V: 2, T: 25000}, {V: 2, T: 30000}}, | 
					
						
							|  |  |  | 							Metric: labels.FromStrings("__name__", "metric")}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Start: time.Unix(35, 0), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					Query: "metric[20s:5s] offset 6s", | 
					
						
							|  |  |  | 					Result: Result{ | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 						Matrix{Series{ | 
					
						
							|  |  |  | 							Points: []Point{{V: 2, T: 10000}, {V: 2, T: 15000}, {V: 2, T: 20000}, {V: 2, T: 25000}}, | 
					
						
							|  |  |  | 							Metric: labels.FromStrings("__name__", "metric")}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Start: time.Unix(35, 0), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					Query: "metric[20s:5s] offset 7s", | 
					
						
							|  |  |  | 					Result: Result{ | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 						Matrix{Series{ | 
					
						
							|  |  |  | 							Points: []Point{{V: 2, T: 10000}, {V: 2, T: 15000}, {V: 2, T: 20000}, {V: 2, T: 25000}}, | 
					
						
							|  |  |  | 							Metric: labels.FromStrings("__name__", "metric")}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Start: time.Unix(35, 0), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			loadString: `load 10s | 
					
						
							|  |  |  | 							http_requests{job="api-server", instance="0", group="production"}	0+10x1000 100+30x1000 | 
					
						
							|  |  |  | 							http_requests{job="api-server", instance="1", group="production"}	0+20x1000 200+30x1000 | 
					
						
							|  |  |  | 							http_requests{job="api-server", instance="0", group="canary"}		0+30x1000 300+80x1000 | 
					
						
							|  |  |  | 							http_requests{job="api-server", instance="1", group="canary"}		0+40x2000`, | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 			cases: []caseType{ | 
					
						
							| 
									
										
										
										
											2018-12-22 21:47:13 +08:00
										 |  |  | 				{ // Normal selector.
 | 
					
						
							|  |  |  | 					Query: `http_requests{group=~"pro.*",instance="0"}[30s:10s]`, | 
					
						
							|  |  |  | 					Result: Result{ | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 						Matrix{Series{ | 
					
						
							|  |  |  | 							Points: []Point{{V: 9990, T: 9990000}, {V: 10000, T: 10000000}, {V: 100, T: 10010000}, {V: 130, T: 10020000}}, | 
					
						
							|  |  |  | 							Metric: labels.FromStrings("__name__", "http_requests", "job", "api-server", "instance", "0", "group", "production")}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Start: time.Unix(10020, 0), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				{ // Default step.
 | 
					
						
							|  |  |  | 					Query: `http_requests{group=~"pro.*",instance="0"}[5m:]`, | 
					
						
							|  |  |  | 					Result: Result{ | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 						Matrix{Series{ | 
					
						
							|  |  |  | 							Points: []Point{{V: 9840, T: 9840000}, {V: 9900, T: 9900000}, {V: 9960, T: 9960000}, {V: 130, T: 10020000}, {V: 310, T: 10080000}}, | 
					
						
							|  |  |  | 							Metric: labels.FromStrings("__name__", "http_requests", "job", "api-server", "instance", "0", "group", "production")}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Start: time.Unix(10100, 0), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				{ // Checking if high offset (>LookbackDelta) is being taken care of.
 | 
					
						
							|  |  |  | 					Query: `http_requests{group=~"pro.*",instance="0"}[5m:] offset 20m`, | 
					
						
							|  |  |  | 					Result: Result{ | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 						Matrix{Series{ | 
					
						
							|  |  |  | 							Points: []Point{{V: 8640, T: 8640000}, {V: 8700, T: 8700000}, {V: 8760, T: 8760000}, {V: 8820, T: 8820000}, {V: 8880, T: 8880000}}, | 
					
						
							|  |  |  | 							Metric: labels.FromStrings("__name__", "http_requests", "job", "api-server", "instance", "0", "group", "production")}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Start: time.Unix(10100, 0), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					Query: `rate(http_requests[1m])[15s:5s]`, | 
					
						
							|  |  |  | 					Result: Result{ | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 						Matrix{ | 
					
						
							|  |  |  | 							Series{ | 
					
						
							|  |  |  | 								Points: []Point{{V: 3, T: 7985000}, {V: 3, T: 7990000}, {V: 3, T: 7995000}, {V: 3, T: 8000000}}, | 
					
						
							|  |  |  | 								Metric: labels.FromStrings("job", "api-server", "instance", "0", "group", "canary"), | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 							Series{ | 
					
						
							|  |  |  | 								Points: []Point{{V: 4, T: 7985000}, {V: 4, T: 7990000}, {V: 4, T: 7995000}, {V: 4, T: 8000000}}, | 
					
						
							|  |  |  | 								Metric: labels.FromStrings("job", "api-server", "instance", "1", "group", "canary"), | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 							Series{ | 
					
						
							|  |  |  | 								Points: []Point{{V: 1, T: 7985000}, {V: 1, T: 7990000}, {V: 1, T: 7995000}, {V: 1, T: 8000000}}, | 
					
						
							|  |  |  | 								Metric: labels.FromStrings("job", "api-server", "instance", "0", "group", "production"), | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 							Series{ | 
					
						
							|  |  |  | 								Points: []Point{{V: 2, T: 7985000}, {V: 2, T: 7990000}, {V: 2, T: 7995000}, {V: 2, T: 8000000}}, | 
					
						
							|  |  |  | 								Metric: labels.FromStrings("job", "api-server", "instance", "1", "group", "production"), | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Start: time.Unix(8000, 0), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					Query: `sum(http_requests{group=~"pro.*"})[30s:10s]`, | 
					
						
							|  |  |  | 					Result: Result{ | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 						Matrix{Series{ | 
					
						
							|  |  |  | 							Points: []Point{{V: 270, T: 90000}, {V: 300, T: 100000}, {V: 330, T: 110000}, {V: 360, T: 120000}}, | 
					
						
							|  |  |  | 							Metric: labels.Labels{}}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Start: time.Unix(120, 0), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					Query: `sum(http_requests)[40s:10s]`, | 
					
						
							|  |  |  | 					Result: Result{ | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 						Matrix{Series{ | 
					
						
							|  |  |  | 							Points: []Point{{V: 800, T: 80000}, {V: 900, T: 90000}, {V: 1000, T: 100000}, {V: 1100, T: 110000}, {V: 1200, T: 120000}}, | 
					
						
							|  |  |  | 							Metric: labels.Labels{}}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Start: time.Unix(120, 0), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					Query: `(sum(http_requests{group=~"p.*"})+sum(http_requests{group=~"c.*"}))[20s:5s]`, | 
					
						
							|  |  |  | 					Result: Result{ | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 						Matrix{Series{ | 
					
						
							|  |  |  | 							Points: []Point{{V: 1000, T: 100000}, {V: 1000, T: 105000}, {V: 1100, T: 110000}, {V: 1100, T: 115000}, {V: 1200, T: 120000}}, | 
					
						
							|  |  |  | 							Metric: labels.Labels{}}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						nil, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Start: time.Unix(120, 0), | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, | 
					
						
							| 
									
										
										
										
											2020-07-24 06:05:43 +08:00
										 |  |  | 	} { | 
					
						
							| 
									
										
										
										
											2020-07-22 21:39:51 +08:00
										 |  |  | 		t.Run("", func(t *testing.T) { | 
					
						
							|  |  |  | 			test, err := NewTest(t, tst.loadString) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 			require.NoError(t, err) | 
					
						
							| 
									
										
										
										
											2020-07-22 21:39:51 +08:00
										 |  |  | 			defer test.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 			require.NoError(t, test.Run()) | 
					
						
							| 
									
										
										
										
											2020-07-22 21:39:51 +08:00
										 |  |  | 			engine := test.QueryEngine() | 
					
						
							|  |  |  | 			for _, c := range tst.cases { | 
					
						
							|  |  |  | 				t.Run(c.Query, func(t *testing.T) { | 
					
						
							|  |  |  | 					qry, err := engine.NewInstantQuery(test.Queryable(), c.Query, c.Start) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 					require.NoError(t, err) | 
					
						
							| 
									
										
										
										
											2020-07-22 21:39:51 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					res := qry.Exec(test.Context()) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 					require.Equal(t, c.Result.Err, res.Err) | 
					
						
							| 
									
										
										
										
											2020-07-22 21:39:51 +08:00
										 |  |  | 					mat := res.Value.(Matrix) | 
					
						
							|  |  |  | 					sort.Sort(mat) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 					require.Equal(t, c.Result.Value, mat) | 
					
						
							| 
									
										
										
										
											2020-07-22 21:39:51 +08:00
										 |  |  | 				}) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2018-12-22 21:47:13 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | type FakeQueryLogger struct { | 
					
						
							|  |  |  | 	closed bool | 
					
						
							|  |  |  | 	logs   []interface{} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func NewFakeQueryLogger() *FakeQueryLogger { | 
					
						
							|  |  |  | 	return &FakeQueryLogger{ | 
					
						
							|  |  |  | 		closed: false, | 
					
						
							|  |  |  | 		logs:   make([]interface{}, 0), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (f *FakeQueryLogger) Close() error { | 
					
						
							|  |  |  | 	f.closed = true | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (f *FakeQueryLogger) Log(l ...interface{}) error { | 
					
						
							|  |  |  | 	f.logs = append(f.logs, l...) | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestQueryLogger_basic(t *testing.T) { | 
					
						
							|  |  |  | 	opts := EngineOpts{ | 
					
						
							| 
									
										
										
										
											2020-01-29 04:38:49 +08:00
										 |  |  | 		Logger:     nil, | 
					
						
							|  |  |  | 		Reg:        nil, | 
					
						
							|  |  |  | 		MaxSamples: 10, | 
					
						
							|  |  |  | 		Timeout:    10 * time.Second, | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	engine := NewEngine(opts) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	queryExec := func() { | 
					
						
							|  |  |  | 		ctx, cancelCtx := context.WithCancel(context.Background()) | 
					
						
							|  |  |  | 		defer cancelCtx() | 
					
						
							|  |  |  | 		query := engine.newTestQuery(func(ctx context.Context) error { | 
					
						
							|  |  |  | 			return contextDone(ctx, "test statement execution") | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 		res := query.Exec(ctx) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 		require.NoError(t, res.Err) | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-17 19:45:11 +08:00
										 |  |  | 	// Query works without query log initialized.
 | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 	queryExec() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	f1 := NewFakeQueryLogger() | 
					
						
							|  |  |  | 	engine.SetQueryLogger(f1) | 
					
						
							|  |  |  | 	queryExec() | 
					
						
							| 
									
										
										
										
											2020-01-27 17:53:10 +08:00
										 |  |  | 	for i, field := range []interface{}{"params", map[string]interface{}{"query": "test statement"}} { | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 		require.Equal(t, field, f1.logs[i]) | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	l := len(f1.logs) | 
					
						
							|  |  |  | 	queryExec() | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.Equal(t, 2*l, len(f1.logs)) | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Test that we close the query logger when unsetting it.
 | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.False(t, f1.closed, "expected f1 to be open, got closed") | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 	engine.SetQueryLogger(nil) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.True(t, f1.closed, "expected f1 to be closed, got open") | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 	queryExec() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Test that we close the query logger when swapping.
 | 
					
						
							|  |  |  | 	f2 := NewFakeQueryLogger() | 
					
						
							|  |  |  | 	f3 := NewFakeQueryLogger() | 
					
						
							|  |  |  | 	engine.SetQueryLogger(f2) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.False(t, f2.closed, "expected f2 to be open, got closed") | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 	queryExec() | 
					
						
							|  |  |  | 	engine.SetQueryLogger(f3) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.True(t, f2.closed, "expected f2 to be closed, got open") | 
					
						
							|  |  |  | 	require.False(t, f3.closed, "expected f3 to be open, got closed") | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 	queryExec() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestQueryLogger_fields(t *testing.T) { | 
					
						
							|  |  |  | 	opts := EngineOpts{ | 
					
						
							| 
									
										
										
										
											2020-01-29 04:38:49 +08:00
										 |  |  | 		Logger:     nil, | 
					
						
							|  |  |  | 		Reg:        nil, | 
					
						
							|  |  |  | 		MaxSamples: 10, | 
					
						
							|  |  |  | 		Timeout:    10 * time.Second, | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	engine := NewEngine(opts) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	f1 := NewFakeQueryLogger() | 
					
						
							|  |  |  | 	engine.SetQueryLogger(f1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ctx, cancelCtx := context.WithCancel(context.Background()) | 
					
						
							| 
									
										
										
										
											2020-01-27 17:53:10 +08:00
										 |  |  | 	ctx = NewOriginContext(ctx, map[string]interface{}{"foo": "bar"}) | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 	defer cancelCtx() | 
					
						
							|  |  |  | 	query := engine.newTestQuery(func(ctx context.Context) error { | 
					
						
							|  |  |  | 		return contextDone(ctx, "test statement execution") | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	res := query.Exec(ctx) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.NoError(t, res.Err) | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	expected := []string{"foo", "bar"} | 
					
						
							|  |  |  | 	for i, field := range expected { | 
					
						
							|  |  |  | 		v := f1.logs[len(f1.logs)-len(expected)+i].(string) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 		require.Equal(t, field, v) | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestQueryLogger_error(t *testing.T) { | 
					
						
							|  |  |  | 	opts := EngineOpts{ | 
					
						
							| 
									
										
										
										
											2020-01-29 04:38:49 +08:00
										 |  |  | 		Logger:     nil, | 
					
						
							|  |  |  | 		Reg:        nil, | 
					
						
							|  |  |  | 		MaxSamples: 10, | 
					
						
							|  |  |  | 		Timeout:    10 * time.Second, | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	engine := NewEngine(opts) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	f1 := NewFakeQueryLogger() | 
					
						
							|  |  |  | 	engine.SetQueryLogger(f1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ctx, cancelCtx := context.WithCancel(context.Background()) | 
					
						
							| 
									
										
										
										
											2020-01-27 17:53:10 +08:00
										 |  |  | 	ctx = NewOriginContext(ctx, map[string]interface{}{"foo": "bar"}) | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 	defer cancelCtx() | 
					
						
							|  |  |  | 	testErr := errors.New("failure") | 
					
						
							|  |  |  | 	query := engine.newTestQuery(func(ctx context.Context) error { | 
					
						
							|  |  |  | 		return testErr | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	res := query.Exec(ctx) | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 	require.Error(t, res.Err, "query should have failed") | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-27 17:53:10 +08:00
										 |  |  | 	for i, field := range []interface{}{"params", map[string]interface{}{"query": "test statement"}, "error", testErr} { | 
					
						
							| 
									
										
										
										
											2020-10-29 17:43:23 +08:00
										 |  |  | 		require.Equal(t, f1.logs[i], field) | 
					
						
							| 
									
										
										
										
											2020-01-08 21:28:43 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-10 00:03:16 +08:00
										 |  |  | func TestPreprocessAndWrapWithStepInvariantExpr(t *testing.T) { | 
					
						
							|  |  |  | 	startTime := time.Unix(1000, 0) | 
					
						
							|  |  |  | 	endTime := time.Unix(9999, 0) | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 	var testCases = []struct { | 
					
						
							|  |  |  | 		input    string      // The input to be parsed.
 | 
					
						
							|  |  |  | 		expected parser.Expr // The expected expression AST.
 | 
					
						
							|  |  |  | 	}{ | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			input: "123.4567", | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.NumberLiteral{ | 
					
						
							|  |  |  | 					Val:      123.4567, | 
					
						
							|  |  |  | 					PosRange: parser.PositionRange{Start: 0, End: 8}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: `"foo"`, | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.StringLiteral{ | 
					
						
							|  |  |  | 					Val:      "foo", | 
					
						
							|  |  |  | 					PosRange: parser.PositionRange{Start: 0, End: 5}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: "foo * bar", | 
					
						
							|  |  |  | 			expected: &parser.BinaryExpr{ | 
					
						
							|  |  |  | 				Op: parser.MUL, | 
					
						
							|  |  |  | 				LHS: &parser.VectorSelector{ | 
					
						
							|  |  |  | 					Name: "foo", | 
					
						
							|  |  |  | 					LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 						parser.MustLabelMatcher(labels.MatchEqual, "__name__", "foo"), | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 						Start: 0, | 
					
						
							|  |  |  | 						End:   3, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				RHS: &parser.VectorSelector{ | 
					
						
							|  |  |  | 					Name: "bar", | 
					
						
							|  |  |  | 					LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 						parser.MustLabelMatcher(labels.MatchEqual, "__name__", "bar"), | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 						Start: 6, | 
					
						
							|  |  |  | 						End:   9, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				VectorMatching: &parser.VectorMatching{Card: parser.CardOneToOne}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: "foo * bar @ 10", | 
					
						
							|  |  |  | 			expected: &parser.BinaryExpr{ | 
					
						
							|  |  |  | 				Op: parser.MUL, | 
					
						
							|  |  |  | 				LHS: &parser.VectorSelector{ | 
					
						
							|  |  |  | 					Name: "foo", | 
					
						
							|  |  |  | 					LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 						parser.MustLabelMatcher(labels.MatchEqual, "__name__", "foo"), | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 						Start: 0, | 
					
						
							|  |  |  | 						End:   3, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				RHS: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 					Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 						Name: "bar", | 
					
						
							|  |  |  | 						LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 							parser.MustLabelMatcher(labels.MatchEqual, "__name__", "bar"), | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 6, | 
					
						
							|  |  |  | 							End:   14, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						Timestamp: makeInt64Pointer(10000), | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				VectorMatching: &parser.VectorMatching{Card: parser.CardOneToOne}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: "foo @ 20 * bar @ 10", | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.BinaryExpr{ | 
					
						
							|  |  |  | 					Op: parser.MUL, | 
					
						
							|  |  |  | 					LHS: &parser.VectorSelector{ | 
					
						
							|  |  |  | 						Name: "foo", | 
					
						
							|  |  |  | 						LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 							parser.MustLabelMatcher(labels.MatchEqual, "__name__", "foo"), | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 0, | 
					
						
							|  |  |  | 							End:   8, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						Timestamp: makeInt64Pointer(20000), | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					RHS: &parser.VectorSelector{ | 
					
						
							|  |  |  | 						Name: "bar", | 
					
						
							|  |  |  | 						LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 							parser.MustLabelMatcher(labels.MatchEqual, "__name__", "bar"), | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 11, | 
					
						
							|  |  |  | 							End:   19, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						Timestamp: makeInt64Pointer(10000), | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					VectorMatching: &parser.VectorMatching{Card: parser.CardOneToOne}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: "test[5s]", | 
					
						
							|  |  |  | 			expected: &parser.MatrixSelector{ | 
					
						
							|  |  |  | 				VectorSelector: &parser.VectorSelector{ | 
					
						
							|  |  |  | 					Name: "test", | 
					
						
							|  |  |  | 					LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 						parser.MustLabelMatcher(labels.MatchEqual, "__name__", "test"), | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 						Start: 0, | 
					
						
							|  |  |  | 						End:   4, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				Range:  5 * time.Second, | 
					
						
							|  |  |  | 				EndPos: 8, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: `test{a="b"}[5y] @ 1603774699`, | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.MatrixSelector{ | 
					
						
							|  |  |  | 					VectorSelector: &parser.VectorSelector{ | 
					
						
							|  |  |  | 						Name:      "test", | 
					
						
							|  |  |  | 						Timestamp: makeInt64Pointer(1603774699000), | 
					
						
							|  |  |  | 						LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 							parser.MustLabelMatcher(labels.MatchEqual, "a", "b"), | 
					
						
							|  |  |  | 							parser.MustLabelMatcher(labels.MatchEqual, "__name__", "test"), | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 0, | 
					
						
							|  |  |  | 							End:   11, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Range:  5 * 365 * 24 * time.Hour, | 
					
						
							|  |  |  | 					EndPos: 28, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: "sum by (foo)(some_metric)", | 
					
						
							|  |  |  | 			expected: &parser.AggregateExpr{ | 
					
						
							|  |  |  | 				Op: parser.SUM, | 
					
						
							|  |  |  | 				Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 					Name: "some_metric", | 
					
						
							|  |  |  | 					LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 						parser.MustLabelMatcher(labels.MatchEqual, "__name__", "some_metric"), | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 						Start: 13, | 
					
						
							|  |  |  | 						End:   24, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				Grouping: []string{"foo"}, | 
					
						
							|  |  |  | 				PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 					Start: 0, | 
					
						
							|  |  |  | 					End:   25, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: "sum by (foo)(some_metric @ 10)", | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.AggregateExpr{ | 
					
						
							|  |  |  | 					Op: parser.SUM, | 
					
						
							|  |  |  | 					Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 						Name: "some_metric", | 
					
						
							|  |  |  | 						LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 							parser.MustLabelMatcher(labels.MatchEqual, "__name__", "some_metric"), | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 13, | 
					
						
							|  |  |  | 							End:   29, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						Timestamp: makeInt64Pointer(10000), | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Grouping: []string{"foo"}, | 
					
						
							|  |  |  | 					PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 						Start: 0, | 
					
						
							|  |  |  | 						End:   30, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: "sum(some_metric1 @ 10) + sum(some_metric2 @ 20)", | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.BinaryExpr{ | 
					
						
							|  |  |  | 					Op:             parser.ADD, | 
					
						
							|  |  |  | 					VectorMatching: &parser.VectorMatching{}, | 
					
						
							|  |  |  | 					LHS: &parser.AggregateExpr{ | 
					
						
							|  |  |  | 						Op: parser.SUM, | 
					
						
							|  |  |  | 						Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 							Name: "some_metric1", | 
					
						
							|  |  |  | 							LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 								parser.MustLabelMatcher(labels.MatchEqual, "__name__", "some_metric1"), | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 							PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 								Start: 4, | 
					
						
							|  |  |  | 								End:   21, | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 							Timestamp: makeInt64Pointer(10000), | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 0, | 
					
						
							|  |  |  | 							End:   22, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					RHS: &parser.AggregateExpr{ | 
					
						
							|  |  |  | 						Op: parser.SUM, | 
					
						
							|  |  |  | 						Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 							Name: "some_metric2", | 
					
						
							|  |  |  | 							LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 								parser.MustLabelMatcher(labels.MatchEqual, "__name__", "some_metric2"), | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 							PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 								Start: 29, | 
					
						
							|  |  |  | 								End:   46, | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 							Timestamp: makeInt64Pointer(20000), | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 25, | 
					
						
							|  |  |  | 							End:   47, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: "some_metric and topk(5, rate(some_metric[1m] @ 20))", | 
					
						
							|  |  |  | 			expected: &parser.BinaryExpr{ | 
					
						
							|  |  |  | 				Op: parser.LAND, | 
					
						
							|  |  |  | 				VectorMatching: &parser.VectorMatching{ | 
					
						
							|  |  |  | 					Card: parser.CardManyToMany, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				LHS: &parser.VectorSelector{ | 
					
						
							|  |  |  | 					Name: "some_metric", | 
					
						
							|  |  |  | 					LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 						parser.MustLabelMatcher(labels.MatchEqual, "__name__", "some_metric"), | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 						Start: 0, | 
					
						
							|  |  |  | 						End:   11, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				RHS: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 					Expr: &parser.AggregateExpr{ | 
					
						
							|  |  |  | 						Op: parser.TOPK, | 
					
						
							|  |  |  | 						Expr: &parser.Call{ | 
					
						
							|  |  |  | 							Func: parser.MustGetFunction("rate"), | 
					
						
							|  |  |  | 							Args: parser.Expressions{ | 
					
						
							|  |  |  | 								&parser.MatrixSelector{ | 
					
						
							|  |  |  | 									VectorSelector: &parser.VectorSelector{ | 
					
						
							|  |  |  | 										Name: "some_metric", | 
					
						
							|  |  |  | 										LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 											parser.MustLabelMatcher(labels.MatchEqual, "__name__", "some_metric"), | 
					
						
							|  |  |  | 										}, | 
					
						
							|  |  |  | 										PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 											Start: 29, | 
					
						
							|  |  |  | 											End:   40, | 
					
						
							|  |  |  | 										}, | 
					
						
							|  |  |  | 										Timestamp: makeInt64Pointer(20000), | 
					
						
							|  |  |  | 									}, | 
					
						
							|  |  |  | 									Range:  1 * time.Minute, | 
					
						
							|  |  |  | 									EndPos: 49, | 
					
						
							|  |  |  | 								}, | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 							PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 								Start: 24, | 
					
						
							|  |  |  | 								End:   50, | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						Param: &parser.NumberLiteral{ | 
					
						
							|  |  |  | 							Val: 5, | 
					
						
							|  |  |  | 							PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 								Start: 21, | 
					
						
							|  |  |  | 								End:   22, | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 16, | 
					
						
							|  |  |  | 							End:   51, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: "time()", | 
					
						
							|  |  |  | 			expected: &parser.Call{ | 
					
						
							|  |  |  | 				Func: parser.MustGetFunction("time"), | 
					
						
							|  |  |  | 				Args: parser.Expressions{}, | 
					
						
							|  |  |  | 				PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 					Start: 0, | 
					
						
							|  |  |  | 					End:   6, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: `foo{bar="baz"}[10m:6s]`, | 
					
						
							|  |  |  | 			expected: &parser.SubqueryExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 					Name: "foo", | 
					
						
							|  |  |  | 					LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 						parser.MustLabelMatcher(labels.MatchEqual, "bar", "baz"), | 
					
						
							|  |  |  | 						parser.MustLabelMatcher(labels.MatchEqual, "__name__", "foo"), | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 						Start: 0, | 
					
						
							|  |  |  | 						End:   14, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				Range:  10 * time.Minute, | 
					
						
							|  |  |  | 				Step:   6 * time.Second, | 
					
						
							|  |  |  | 				EndPos: 22, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: `foo{bar="baz"}[10m:6s] @ 10`, | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.SubqueryExpr{ | 
					
						
							|  |  |  | 					Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 						Name: "foo", | 
					
						
							|  |  |  | 						LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 							parser.MustLabelMatcher(labels.MatchEqual, "bar", "baz"), | 
					
						
							|  |  |  | 							parser.MustLabelMatcher(labels.MatchEqual, "__name__", "foo"), | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 0, | 
					
						
							|  |  |  | 							End:   14, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Range:     10 * time.Minute, | 
					
						
							|  |  |  | 					Step:      6 * time.Second, | 
					
						
							|  |  |  | 					Timestamp: makeInt64Pointer(10000), | 
					
						
							|  |  |  | 					EndPos:    27, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { // Even though the subquery is step invariant, the inside is also wrapped separately.
 | 
					
						
							|  |  |  | 			input: `sum(foo{bar="baz"} @ 20)[10m:6s] @ 10`, | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.SubqueryExpr{ | 
					
						
							|  |  |  | 					Expr: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 						Expr: &parser.AggregateExpr{ | 
					
						
							|  |  |  | 							Op: parser.SUM, | 
					
						
							|  |  |  | 							Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 								Name: "foo", | 
					
						
							|  |  |  | 								LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 									parser.MustLabelMatcher(labels.MatchEqual, "bar", "baz"), | 
					
						
							|  |  |  | 									parser.MustLabelMatcher(labels.MatchEqual, "__name__", "foo"), | 
					
						
							|  |  |  | 								}, | 
					
						
							|  |  |  | 								PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 									Start: 4, | 
					
						
							|  |  |  | 									End:   23, | 
					
						
							|  |  |  | 								}, | 
					
						
							|  |  |  | 								Timestamp: makeInt64Pointer(20000), | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 							PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 								Start: 0, | 
					
						
							|  |  |  | 								End:   24, | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Range:     10 * time.Minute, | 
					
						
							|  |  |  | 					Step:      6 * time.Second, | 
					
						
							|  |  |  | 					Timestamp: makeInt64Pointer(10000), | 
					
						
							|  |  |  | 					EndPos:    37, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: `min_over_time(rate(foo{bar="baz"}[2s])[5m:] @ 1603775091)[4m:3s]`, | 
					
						
							|  |  |  | 			expected: &parser.SubqueryExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 					Expr: &parser.Call{ | 
					
						
							|  |  |  | 						Func: parser.MustGetFunction("min_over_time"), | 
					
						
							|  |  |  | 						Args: parser.Expressions{ | 
					
						
							|  |  |  | 							&parser.SubqueryExpr{ | 
					
						
							|  |  |  | 								Expr: &parser.Call{ | 
					
						
							|  |  |  | 									Func: parser.MustGetFunction("rate"), | 
					
						
							|  |  |  | 									Args: parser.Expressions{ | 
					
						
							|  |  |  | 										&parser.MatrixSelector{ | 
					
						
							|  |  |  | 											VectorSelector: &parser.VectorSelector{ | 
					
						
							|  |  |  | 												Name: "foo", | 
					
						
							|  |  |  | 												LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 													parser.MustLabelMatcher(labels.MatchEqual, "bar", "baz"), | 
					
						
							|  |  |  | 													parser.MustLabelMatcher(labels.MatchEqual, "__name__", "foo"), | 
					
						
							|  |  |  | 												}, | 
					
						
							|  |  |  | 												PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 													Start: 19, | 
					
						
							|  |  |  | 													End:   33, | 
					
						
							|  |  |  | 												}, | 
					
						
							|  |  |  | 											}, | 
					
						
							|  |  |  | 											Range:  2 * time.Second, | 
					
						
							|  |  |  | 											EndPos: 37, | 
					
						
							|  |  |  | 										}, | 
					
						
							|  |  |  | 									}, | 
					
						
							|  |  |  | 									PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 										Start: 14, | 
					
						
							|  |  |  | 										End:   38, | 
					
						
							|  |  |  | 									}, | 
					
						
							|  |  |  | 								}, | 
					
						
							|  |  |  | 								Range:     5 * time.Minute, | 
					
						
							|  |  |  | 								Timestamp: makeInt64Pointer(1603775091000), | 
					
						
							|  |  |  | 								EndPos:    56, | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 0, | 
					
						
							|  |  |  | 							End:   57, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				Range:  4 * time.Minute, | 
					
						
							|  |  |  | 				Step:   3 * time.Second, | 
					
						
							|  |  |  | 				EndPos: 64, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: `some_metric @ 123 offset 1m [10m:5s]`, | 
					
						
							|  |  |  | 			expected: &parser.SubqueryExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 					Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 						Name: "some_metric", | 
					
						
							|  |  |  | 						LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 							parser.MustLabelMatcher(labels.MatchEqual, "__name__", "some_metric"), | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 0, | 
					
						
							|  |  |  | 							End:   27, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						Timestamp:      makeInt64Pointer(123000), | 
					
						
							|  |  |  | 						OriginalOffset: 1 * time.Minute, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 				Range:  10 * time.Minute, | 
					
						
							|  |  |  | 				Step:   5 * time.Second, | 
					
						
							|  |  |  | 				EndPos: 36, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: `some_metric[10m:5s] offset 1m @ 123`, | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.SubqueryExpr{ | 
					
						
							|  |  |  | 					Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 						Name: "some_metric", | 
					
						
							|  |  |  | 						LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 							parser.MustLabelMatcher(labels.MatchEqual, "__name__", "some_metric"), | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 0, | 
					
						
							|  |  |  | 							End:   11, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Timestamp:      makeInt64Pointer(123000), | 
					
						
							|  |  |  | 					OriginalOffset: 1 * time.Minute, | 
					
						
							|  |  |  | 					Range:          10 * time.Minute, | 
					
						
							|  |  |  | 					Step:           5 * time.Second, | 
					
						
							|  |  |  | 					EndPos:         35, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: `(foo + bar{nm="val"} @ 1234)[5m:] @ 1603775019`, | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.SubqueryExpr{ | 
					
						
							|  |  |  | 					Expr: &parser.ParenExpr{ | 
					
						
							|  |  |  | 						Expr: &parser.BinaryExpr{ | 
					
						
							|  |  |  | 							Op: parser.ADD, | 
					
						
							|  |  |  | 							VectorMatching: &parser.VectorMatching{ | 
					
						
							|  |  |  | 								Card: parser.CardOneToOne, | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 							LHS: &parser.VectorSelector{ | 
					
						
							|  |  |  | 								Name: "foo", | 
					
						
							|  |  |  | 								LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 									parser.MustLabelMatcher(labels.MatchEqual, "__name__", "foo"), | 
					
						
							|  |  |  | 								}, | 
					
						
							|  |  |  | 								PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 									Start: 1, | 
					
						
							|  |  |  | 									End:   4, | 
					
						
							|  |  |  | 								}, | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 							RHS: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 								Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 									Name: "bar", | 
					
						
							|  |  |  | 									LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 										parser.MustLabelMatcher(labels.MatchEqual, "nm", "val"), | 
					
						
							|  |  |  | 										parser.MustLabelMatcher(labels.MatchEqual, "__name__", "bar"), | 
					
						
							|  |  |  | 									}, | 
					
						
							|  |  |  | 									Timestamp: makeInt64Pointer(1234000), | 
					
						
							|  |  |  | 									PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 										Start: 7, | 
					
						
							|  |  |  | 										End:   27, | 
					
						
							|  |  |  | 									}, | 
					
						
							|  |  |  | 								}, | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 0, | 
					
						
							|  |  |  | 							End:   28, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Range:     5 * time.Minute, | 
					
						
							|  |  |  | 					Timestamp: makeInt64Pointer(1603775019000), | 
					
						
							|  |  |  | 					EndPos:    46, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: "abs(abs(metric @ 10))", | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.Call{ | 
					
						
							|  |  |  | 					Func: &parser.Function{ | 
					
						
							|  |  |  | 						Name:       "abs", | 
					
						
							|  |  |  | 						ArgTypes:   []parser.ValueType{parser.ValueTypeVector}, | 
					
						
							|  |  |  | 						ReturnType: parser.ValueTypeVector, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Args: parser.Expressions{&parser.Call{ | 
					
						
							|  |  |  | 						Func: &parser.Function{ | 
					
						
							|  |  |  | 							Name:       "abs", | 
					
						
							|  |  |  | 							ArgTypes:   []parser.ValueType{parser.ValueTypeVector}, | 
					
						
							|  |  |  | 							ReturnType: parser.ValueTypeVector, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						Args: parser.Expressions{&parser.VectorSelector{ | 
					
						
							|  |  |  | 							Name: "metric", | 
					
						
							|  |  |  | 							LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 								parser.MustLabelMatcher(labels.MatchEqual, "__name__", "metric"), | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 							PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 								Start: 8, | 
					
						
							|  |  |  | 								End:   19, | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 							Timestamp: makeInt64Pointer(10000), | 
					
						
							|  |  |  | 						}}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 4, | 
					
						
							|  |  |  | 							End:   20, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}}, | 
					
						
							|  |  |  | 					PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 						Start: 0, | 
					
						
							|  |  |  | 						End:   21, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: "sum(sum(some_metric1 @ 10) + sum(some_metric2 @ 20))", | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.AggregateExpr{ | 
					
						
							|  |  |  | 					Op: parser.SUM, | 
					
						
							|  |  |  | 					Expr: &parser.BinaryExpr{ | 
					
						
							|  |  |  | 						Op:             parser.ADD, | 
					
						
							|  |  |  | 						VectorMatching: &parser.VectorMatching{}, | 
					
						
							|  |  |  | 						LHS: &parser.AggregateExpr{ | 
					
						
							|  |  |  | 							Op: parser.SUM, | 
					
						
							|  |  |  | 							Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 								Name: "some_metric1", | 
					
						
							|  |  |  | 								LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 									parser.MustLabelMatcher(labels.MatchEqual, "__name__", "some_metric1"), | 
					
						
							|  |  |  | 								}, | 
					
						
							|  |  |  | 								PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 									Start: 8, | 
					
						
							|  |  |  | 									End:   25, | 
					
						
							|  |  |  | 								}, | 
					
						
							|  |  |  | 								Timestamp: makeInt64Pointer(10000), | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 							PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 								Start: 4, | 
					
						
							|  |  |  | 								End:   26, | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						RHS: &parser.AggregateExpr{ | 
					
						
							|  |  |  | 							Op: parser.SUM, | 
					
						
							|  |  |  | 							Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 								Name: "some_metric2", | 
					
						
							|  |  |  | 								LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 									parser.MustLabelMatcher(labels.MatchEqual, "__name__", "some_metric2"), | 
					
						
							|  |  |  | 								}, | 
					
						
							|  |  |  | 								PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 									Start: 33, | 
					
						
							|  |  |  | 									End:   50, | 
					
						
							|  |  |  | 								}, | 
					
						
							|  |  |  | 								Timestamp: makeInt64Pointer(20000), | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 							PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 								Start: 29, | 
					
						
							|  |  |  | 								End:   52, | 
					
						
							|  |  |  | 							}, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 						Start: 0, | 
					
						
							|  |  |  | 						End:   52, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							| 
									
										
										
										
											2021-02-10 00:03:16 +08:00
										 |  |  | 		}, { | 
					
						
							|  |  |  | 			input: `foo @ start()`, | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 					Name: "foo", | 
					
						
							|  |  |  | 					LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 						parser.MustLabelMatcher(labels.MatchEqual, "__name__", "foo"), | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 						Start: 0, | 
					
						
							|  |  |  | 						End:   13, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Timestamp:  makeInt64Pointer(timestamp.FromTime(startTime)), | 
					
						
							|  |  |  | 					StartOrEnd: parser.START, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: `foo @ end()`, | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 					Name: "foo", | 
					
						
							|  |  |  | 					LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 						parser.MustLabelMatcher(labels.MatchEqual, "__name__", "foo"), | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 						Start: 0, | 
					
						
							|  |  |  | 						End:   11, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Timestamp:  makeInt64Pointer(timestamp.FromTime(endTime)), | 
					
						
							|  |  |  | 					StartOrEnd: parser.END, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: `test[5y] @ start()`, | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.MatrixSelector{ | 
					
						
							|  |  |  | 					VectorSelector: &parser.VectorSelector{ | 
					
						
							|  |  |  | 						Name:       "test", | 
					
						
							|  |  |  | 						Timestamp:  makeInt64Pointer(timestamp.FromTime(startTime)), | 
					
						
							|  |  |  | 						StartOrEnd: parser.START, | 
					
						
							|  |  |  | 						LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 							parser.MustLabelMatcher(labels.MatchEqual, "__name__", "test"), | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 0, | 
					
						
							|  |  |  | 							End:   4, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Range:  5 * 365 * 24 * time.Hour, | 
					
						
							|  |  |  | 					EndPos: 18, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: `test[5y] @ end()`, | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.MatrixSelector{ | 
					
						
							|  |  |  | 					VectorSelector: &parser.VectorSelector{ | 
					
						
							|  |  |  | 						Name:       "test", | 
					
						
							|  |  |  | 						Timestamp:  makeInt64Pointer(timestamp.FromTime(endTime)), | 
					
						
							|  |  |  | 						StartOrEnd: parser.END, | 
					
						
							|  |  |  | 						LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 							parser.MustLabelMatcher(labels.MatchEqual, "__name__", "test"), | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 0, | 
					
						
							|  |  |  | 							End:   4, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Range:  5 * 365 * 24 * time.Hour, | 
					
						
							|  |  |  | 					EndPos: 16, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: `some_metric[10m:5s] @ start()`, | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.SubqueryExpr{ | 
					
						
							|  |  |  | 					Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 						Name: "some_metric", | 
					
						
							|  |  |  | 						LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 							parser.MustLabelMatcher(labels.MatchEqual, "__name__", "some_metric"), | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 0, | 
					
						
							|  |  |  | 							End:   11, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Timestamp:  makeInt64Pointer(timestamp.FromTime(startTime)), | 
					
						
							|  |  |  | 					StartOrEnd: parser.START, | 
					
						
							|  |  |  | 					Range:      10 * time.Minute, | 
					
						
							|  |  |  | 					Step:       5 * time.Second, | 
					
						
							|  |  |  | 					EndPos:     29, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			input: `some_metric[10m:5s] @ end()`, | 
					
						
							|  |  |  | 			expected: &parser.StepInvariantExpr{ | 
					
						
							|  |  |  | 				Expr: &parser.SubqueryExpr{ | 
					
						
							|  |  |  | 					Expr: &parser.VectorSelector{ | 
					
						
							|  |  |  | 						Name: "some_metric", | 
					
						
							|  |  |  | 						LabelMatchers: []*labels.Matcher{ | 
					
						
							|  |  |  | 							parser.MustLabelMatcher(labels.MatchEqual, "__name__", "some_metric"), | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 						PosRange: parser.PositionRange{ | 
					
						
							|  |  |  | 							Start: 0, | 
					
						
							|  |  |  | 							End:   11, | 
					
						
							|  |  |  | 						}, | 
					
						
							|  |  |  | 					}, | 
					
						
							|  |  |  | 					Timestamp:  makeInt64Pointer(timestamp.FromTime(endTime)), | 
					
						
							|  |  |  | 					StartOrEnd: parser.END, | 
					
						
							|  |  |  | 					Range:      10 * time.Minute, | 
					
						
							|  |  |  | 					Step:       5 * time.Second, | 
					
						
							|  |  |  | 					EndPos:     27, | 
					
						
							|  |  |  | 				}, | 
					
						
							|  |  |  | 			}, | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, test := range testCases { | 
					
						
							|  |  |  | 		t.Run(test.input, func(t *testing.T) { | 
					
						
							|  |  |  | 			expr, err := parser.ParseExpr(test.input) | 
					
						
							|  |  |  | 			require.NoError(t, err) | 
					
						
							| 
									
										
										
										
											2021-02-10 00:03:16 +08:00
										 |  |  | 			expr = PreprocessExpr(expr, startTime, endTime) | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 			require.Equal(t, test.expected, expr, "error on input '%s'", test.input) | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestEngineOptsValidation(t *testing.T) { | 
					
						
							|  |  |  | 	cases := []struct { | 
					
						
							|  |  |  | 		opts     EngineOpts | 
					
						
							|  |  |  | 		query    string | 
					
						
							|  |  |  | 		fail     bool | 
					
						
							| 
									
										
										
										
											2021-02-10 00:03:16 +08:00
										 |  |  | 		expError error | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 	}{ | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2021-02-10 00:03:16 +08:00
										 |  |  | 			opts:  EngineOpts{EnableAtModifier: false}, | 
					
						
							|  |  |  | 			query: "metric @ 100", fail: true, expError: ErrValidationAtModifierDisabled, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			opts:  EngineOpts{EnableAtModifier: false}, | 
					
						
							|  |  |  | 			query: "rate(metric[1m] @ 100)", fail: true, expError: ErrValidationAtModifierDisabled, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			opts:  EngineOpts{EnableAtModifier: false}, | 
					
						
							|  |  |  | 			query: "rate(metric[1h:1m] @ 100)", fail: true, expError: ErrValidationAtModifierDisabled, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			opts:  EngineOpts{EnableAtModifier: false}, | 
					
						
							|  |  |  | 			query: "metric @ start()", fail: true, expError: ErrValidationAtModifierDisabled, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			opts:  EngineOpts{EnableAtModifier: false}, | 
					
						
							|  |  |  | 			query: "rate(metric[1m] @ start())", fail: true, expError: ErrValidationAtModifierDisabled, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			opts:  EngineOpts{EnableAtModifier: false}, | 
					
						
							|  |  |  | 			query: "rate(metric[1h:1m] @ start())", fail: true, expError: ErrValidationAtModifierDisabled, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			opts:  EngineOpts{EnableAtModifier: false}, | 
					
						
							|  |  |  | 			query: "metric @ end()", fail: true, expError: ErrValidationAtModifierDisabled, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			opts:  EngineOpts{EnableAtModifier: false}, | 
					
						
							|  |  |  | 			query: "rate(metric[1m] @ end())", fail: true, expError: ErrValidationAtModifierDisabled, | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			opts:  EngineOpts{EnableAtModifier: false}, | 
					
						
							|  |  |  | 			query: "rate(metric[1h:1m] @ end())", fail: true, expError: ErrValidationAtModifierDisabled, | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 		}, { | 
					
						
							|  |  |  | 			opts:  EngineOpts{EnableAtModifier: true}, | 
					
						
							|  |  |  | 			query: "metric @ 100", | 
					
						
							| 
									
										
										
										
											2021-02-10 00:03:16 +08:00
										 |  |  | 		}, { | 
					
						
							|  |  |  | 			opts:  EngineOpts{EnableAtModifier: true}, | 
					
						
							|  |  |  | 			query: "rate(metric[1m] @ start())", | 
					
						
							|  |  |  | 		}, { | 
					
						
							|  |  |  | 			opts:  EngineOpts{EnableAtModifier: true}, | 
					
						
							|  |  |  | 			query: "rate(metric[1h:1m] @ end())", | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, c := range cases { | 
					
						
							|  |  |  | 		eng := NewEngine(c.opts) | 
					
						
							|  |  |  | 		_, err1 := eng.NewInstantQuery(nil, c.query, time.Unix(10, 0)) | 
					
						
							|  |  |  | 		_, err2 := eng.NewRangeQuery(nil, c.query, time.Unix(0, 0), time.Unix(10, 0), time.Second) | 
					
						
							|  |  |  | 		if c.fail { | 
					
						
							| 
									
										
										
										
											2021-02-10 00:03:16 +08:00
										 |  |  | 			require.Equal(t, c.expError, err1) | 
					
						
							|  |  |  | 			require.Equal(t, c.expError, err2) | 
					
						
							| 
									
										
										
										
											2021-01-20 18:57:39 +08:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			require.Nil(t, err1) | 
					
						
							|  |  |  | 			require.Nil(t, err2) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |