TestQueryStateInfoResource.java
/*
* 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.
*/
package com.facebook.presto.server;
import com.facebook.airlift.http.client.HttpClient;
import com.facebook.airlift.http.client.Request;
import com.facebook.airlift.http.client.UnexpectedResponseException;
import com.facebook.airlift.http.client.jetty.JettyHttpClient;
import com.facebook.airlift.json.JsonCodec;
import com.facebook.presto.client.QueryResults;
import com.facebook.presto.server.testing.TestingPrestoServer;
import com.facebook.presto.tpch.TpchPlugin;
import org.testng.annotations.AfterClass;
import org.testng.annotations.BeforeClass;
import org.testng.annotations.Test;
import java.util.List;
import static com.facebook.airlift.http.client.HttpUriBuilder.uriBuilderFrom;
import static com.facebook.airlift.http.client.JsonResponseHandler.createJsonResponseHandler;
import static com.facebook.airlift.http.client.Request.Builder.prepareGet;
import static com.facebook.airlift.http.client.Request.Builder.preparePost;
import static com.facebook.airlift.http.client.StaticBodyGenerator.createStaticBodyGenerator;
import static com.facebook.airlift.json.JsonCodec.jsonCodec;
import static com.facebook.airlift.json.JsonCodec.listJsonCodec;
import static com.facebook.airlift.testing.Closeables.closeQuietly;
import static com.facebook.presto.client.PrestoHeaders.PRESTO_USER;
import static com.facebook.presto.execution.QueryState.RUNNING;
import static com.facebook.presto.testing.assertions.Assert.assertEquals;
import static java.nio.charset.StandardCharsets.UTF_8;
import static org.testng.Assert.assertNotNull;
import static org.testng.Assert.assertThrows;
import static org.testng.Assert.assertTrue;
@Test(singleThreaded = true)
public class TestQueryStateInfoResource
{
private static final String LONG_LASTING_QUERY = "SELECT * FROM tpch.sf1.lineitem";
private static final JsonCodec<QueryResults> QUERY_RESULTS_JSON_CODEC = jsonCodec(QueryResults.class);
private TestingPrestoServer server;
private HttpClient client;
private QueryResults queryResults;
TestQueryStateInfoResource()
throws Exception
{
server = new TestingPrestoServer();
server.installPlugin(new TpchPlugin());
server.createCatalog("tpch", "tpch");
client = new JettyHttpClient();
}
@BeforeClass
public void setup()
{
Request request1 = preparePost()
.setUri(uriBuilderFrom(server.getBaseUrl()).replacePath("/v1/statement").build())
.setBodyGenerator(createStaticBodyGenerator(LONG_LASTING_QUERY, UTF_8))
.setHeader(PRESTO_USER, "user1")
.build();
queryResults = client.execute(request1, createJsonResponseHandler(QUERY_RESULTS_JSON_CODEC));
client.execute(prepareGet().setUri(queryResults.getNextUri()).build(), createJsonResponseHandler(QUERY_RESULTS_JSON_CODEC));
Request request2 = preparePost()
.setUri(uriBuilderFrom(server.getBaseUrl()).replacePath("/v1/statement").build())
.setBodyGenerator(createStaticBodyGenerator(LONG_LASTING_QUERY, UTF_8))
.setHeader(PRESTO_USER, "user2")
.build();
QueryResults queryResults2 = client.execute(request2, createJsonResponseHandler(jsonCodec(QueryResults.class)));
client.execute(prepareGet().setUri(queryResults2.getNextUri()).build(), createJsonResponseHandler(QUERY_RESULTS_JSON_CODEC));
Request request3 = preparePost()
.setUri(uriBuilderFrom(server.getBaseUrl()).replacePath("/v1/statement").build())
.setBodyGenerator(createStaticBodyGenerator(LONG_LASTING_QUERY, UTF_8))
.setHeader(PRESTO_USER, "alt3")
.build();
QueryResults queryResults3 = client.execute(request3, createJsonResponseHandler(jsonCodec(QueryResults.class)));
client.execute(prepareGet().setUri(queryResults3.getNextUri()).build(), createJsonResponseHandler(QUERY_RESULTS_JSON_CODEC));
// queries are started in the background, so they may not all be immediately visible
while (true) {
List<BasicQueryInfo> queryInfos = client.execute(
prepareGet().setUri(uriBuilderFrom(server.getBaseUrl()).replacePath("/v1/query").build()).build(),
createJsonResponseHandler(listJsonCodec(BasicQueryInfo.class)));
if ((queryInfos.size() == 3) && queryInfos.stream().allMatch(info -> info.getState() == RUNNING)) {
break;
}
}
}
@AfterClass(alwaysRun = true)
public void teardown()
{
closeQuietly(server);
closeQuietly(client);
}
@Test
public void testGetAllQueryStateInfos()
{
List<QueryStateInfo> infos = client.execute(
prepareGet().setUri(server.resolve("/v1/queryState")).build(),
createJsonResponseHandler(listJsonCodec(QueryStateInfo.class)));
assertEquals(infos.size(), 3);
}
@Test
public void testGetQueryStateInfosForUser()
{
List<QueryStateInfo> infos = client.execute(
prepareGet().setUri(server.resolve("/v1/queryState?user=user2")).build(),
createJsonResponseHandler(listJsonCodec(QueryStateInfo.class)));
assertEquals(infos.size(), 1);
}
@Test
public void testGetQueryStateInfosForUserNoResult()
{
List<QueryStateInfo> infos = client.execute(
prepareGet().setUri(server.resolve("/v1/queryState?user=user3")).build(),
createJsonResponseHandler(listJsonCodec(QueryStateInfo.class)));
assertTrue(infos.isEmpty());
}
@Test
public void testGetQueryStateInfo()
{
QueryStateInfo info = client.execute(
prepareGet().setUri(server.resolve("/v1/queryState/" + queryResults.getId())).build(),
createJsonResponseHandler(jsonCodec(QueryStateInfo.class)));
assertNotNull(info);
}
@Test
public void testQueryStateInfoRegexMatching()
{
// Match strings containing "user"
List<QueryStateInfo> infos = client.execute(
prepareGet()
.setUri(server.resolve("/v1/queryState?user=user%5Cd")) // Encoded user\d
.build(),
createJsonResponseHandler(listJsonCodec(QueryStateInfo.class)));
assertEquals(infos.size(), 2);
// Match strings ending in a digit
infos = client.execute(
prepareGet()
.setUri(server.resolve("/v1/queryState?user=%2E%2A%5Cd")) // Encoded .*\d
.build(),
createJsonResponseHandler(listJsonCodec(QueryStateInfo.class)));
assertEquals(infos.size(), 3);
// Non supported version of previous pattern
infos = client.execute(
prepareGet()
.setUri(server.resolve("/v1/queryState?user=%2E%7B%7D%5Cd")) // Encoded .{}\d
.build(),
createJsonResponseHandler(listJsonCodec(QueryStateInfo.class)));
assertEquals(infos.size(), 0);
Request failingReq = prepareGet().setUri(server.resolve("/v1/queryState?user=%2A%5Cd")).build(); // Encoded *\d
assertThrows(() -> client.execute(failingReq, createJsonResponseHandler(listJsonCodec(QueryStateInfo.class))));
}
@Test(expectedExceptions = {UnexpectedResponseException.class}, expectedExceptionsMessageRegExp = "Expected response code .*, but was 404")
public void testGetQueryStateInfoNo()
{
client.execute(
prepareGet().setUri(server.resolve("/v1/queryState/123")).build(),
createJsonResponseHandler(jsonCodec(QueryStateInfo.class)));
}
@Test
public void testGetQueryStateInfoStateFilter()
{
List<QueryStateInfo> infos = client.execute(
prepareGet()
.setUri(server.resolve("/v1/queryState?state=DISPATCHING")) // Encoded user\d
.build(),
createJsonResponseHandler(listJsonCodec(QueryStateInfo.class)));
assertEquals(infos.size(), 0);
infos = client.execute(
prepareGet()
.setUri(server.resolve("/v1/queryState?state=QUEUED"))
.build(),
createJsonResponseHandler(listJsonCodec(QueryStateInfo.class)));
assertEquals(infos.size(), 0);
infos = client.execute(
prepareGet()
.setUri(server.resolve("/v1/queryState?includeAllQueries=true"))
.build(),
createJsonResponseHandler(listJsonCodec(QueryStateInfo.class)));
assertEquals(infos.size(), 3);
}
@Test(expectedExceptions = {UnexpectedResponseException.class}, expectedExceptionsMessageRegExp = "Expected response code .*, but was 400")
public void testGetQueryStateInfoBadStateFilter()
{
client.execute(
prepareGet().setUri(server.resolve("/v1/queryState?state=INVALID_STATE")).build(),
createJsonResponseHandler(jsonCodec(QueryStateInfo.class)));
}
}