/* * SPDX-License-Identifier: Apache-2.0 * * The OpenSearch Contributors require contributions made to * this file be licensed under the Apache-2.0 license or a * compatible open source license. */ /* * Licensed to Elasticsearch under one or more contributor * license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright * ownership. Elasticsearch licenses this file to you 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. */ /* * Modifications Copyright OpenSearch Contributors. See * GitHub history for details. */ package org.opensearch.tasks; import org.opensearch.client.Requests; import org.opensearch.core.common.bytes.BytesReference; import org.opensearch.core.common.io.stream.NamedWriteableRegistry; import org.opensearch.core.common.io.stream.Writeable; import org.opensearch.core.xcontent.XContentBuilder; import org.opensearch.core.xcontent.XContentParser; import org.opensearch.test.AbstractSerializingTestCase; import java.io.IOException; import java.util.Collections; import java.util.HashMap; import java.util.Map; import java.util.function.Predicate; public class TaskInfoTests extends AbstractSerializingTestCase<TaskInfo> { @Override protected TaskInfo doParseInstance(XContentParser parser) { return TaskInfo.fromXContent(parser); } @Override protected TaskInfo createTestInstance() { return randomTaskInfo(); } @Override protected Writeable.Reader<TaskInfo> instanceReader() { return TaskInfo::new; } @Override protected NamedWriteableRegistry getNamedWriteableRegistry() { return new NamedWriteableRegistry( Collections.singletonList(new NamedWriteableRegistry.Entry(Task.Status.class, RawTaskStatus.NAME, RawTaskStatus::new)) ); } @Override protected boolean supportsUnknownFields() { return true; } @Override protected Predicate<String> getRandomFieldsExcludeFilter() { // status, headers and resource_stats hold arbitrary content, we can't inject random fields in them return field -> "status".equals(field) || "headers".equals(field) || field.contains("resource_stats"); } @Override protected TaskInfo mutateInstance(TaskInfo info) { switch (between(0, 11)) { case 0: TaskId taskId = new TaskId(info.getTaskId().getNodeId() + randomAlphaOfLength(5), info.getTaskId().getId()); return new TaskInfo( taskId, info.getType(), info.getAction(), info.getDescription(), info.getStatus(), info.getStartTime(), info.getRunningTimeNanos(), info.isCancellable(), info.isCancelled(), info.getParentTaskId(), info.getHeaders(), info.getResourceStats(), info.getCancellationStartTime() ); case 1: return new TaskInfo( info.getTaskId(), info.getType() + randomAlphaOfLength(5), info.getAction(), info.getDescription(), info.getStatus(), info.getStartTime(), info.getRunningTimeNanos(), info.isCancellable(), info.isCancelled(), info.getParentTaskId(), info.getHeaders(), info.getResourceStats(), info.getCancellationStartTime() ); case 2: return new TaskInfo( info.getTaskId(), info.getType(), info.getAction() + randomAlphaOfLength(5), info.getDescription(), info.getStatus(), info.getStartTime(), info.getRunningTimeNanos(), info.isCancellable(), info.isCancelled(), info.getParentTaskId(), info.getHeaders(), info.getResourceStats(), info.getCancellationStartTime() ); case 3: return new TaskInfo( info.getTaskId(), info.getType(), info.getAction(), info.getDescription() + randomAlphaOfLength(5), info.getStatus(), info.getStartTime(), info.getRunningTimeNanos(), info.isCancellable(), info.isCancelled(), info.getParentTaskId(), info.getHeaders(), info.getResourceStats(), info.getCancellationStartTime() ); case 4: Task.Status newStatus = randomValueOtherThan(info.getStatus(), TaskInfoTests::randomRawTaskStatus); return new TaskInfo( info.getTaskId(), info.getType(), info.getAction(), info.getDescription(), newStatus, info.getStartTime(), info.getRunningTimeNanos(), info.isCancellable(), info.isCancelled(), info.getParentTaskId(), info.getHeaders(), info.getResourceStats(), info.getCancellationStartTime() ); case 5: return new TaskInfo( info.getTaskId(), info.getType(), info.getAction(), info.getDescription(), info.getStatus(), info.getStartTime() + between(1, 100), info.getRunningTimeNanos(), info.isCancellable(), info.isCancelled(), info.getParentTaskId(), info.getHeaders(), info.getResourceStats(), info.getCancellationStartTime() ); case 6: return new TaskInfo( info.getTaskId(), info.getType(), info.getAction(), info.getDescription(), info.getStatus(), info.getStartTime(), info.getRunningTimeNanos() + between(1, 100), info.isCancellable(), info.isCancelled(), info.getParentTaskId(), info.getHeaders(), info.getResourceStats(), info.getCancellationStartTime() ); case 7: return new TaskInfo( info.getTaskId(), info.getType(), info.getAction(), info.getDescription(), info.getStatus(), info.getStartTime(), info.getRunningTimeNanos(), info.isCancellable() == false, false, info.getParentTaskId(), info.getHeaders(), info.getResourceStats(), info.getCancellationStartTime() ); case 8: TaskId parentId = new TaskId(info.getParentTaskId().getNodeId() + randomAlphaOfLength(5), info.getParentTaskId().getId()); return new TaskInfo( info.getTaskId(), info.getType(), info.getAction(), info.getDescription(), info.getStatus(), info.getStartTime(), info.getRunningTimeNanos(), info.isCancellable(), info.isCancelled(), parentId, info.getHeaders(), info.getResourceStats(), info.getCancellationStartTime() ); case 9: Map<String, String> headers = info.getHeaders(); if (headers == null) { headers = new HashMap<>(1); } else { headers = new HashMap<>(info.getHeaders()); } headers.put(randomAlphaOfLength(15), randomAlphaOfLength(15)); return new TaskInfo( info.getTaskId(), info.getType(), info.getAction(), info.getDescription(), info.getStatus(), info.getStartTime(), info.getRunningTimeNanos(), info.isCancellable(), info.isCancelled(), info.getParentTaskId(), headers, info.getResourceStats(), info.getCancellationStartTime() ); case 10: Map<String, TaskResourceUsage> resourceUsageMap; if (info.getResourceStats() == null) { resourceUsageMap = new HashMap<>(1); } else { resourceUsageMap = new HashMap<>(info.getResourceStats().getResourceUsageInfo()); } resourceUsageMap.put(randomAlphaOfLength(5), new TaskResourceUsage(randomNonNegativeLong(), randomNonNegativeLong())); return new TaskInfo( info.getTaskId(), info.getType(), info.getAction(), info.getDescription(), info.getStatus(), info.getStartTime(), info.getRunningTimeNanos(), info.isCancellable(), info.isCancelled(), info.getParentTaskId(), info.getHeaders(), new TaskResourceStats(resourceUsageMap, new TaskThreadUsage(randomInt(10), randomInt(10))) ); case 11: return new TaskInfo( info.getTaskId(), info.getType(), info.getAction(), info.getDescription(), info.getStatus(), info.getStartTime(), info.getRunningTimeNanos(), true, true, info.getParentTaskId(), info.getHeaders(), info.getResourceStats(), randomNonNegativeLong() ); default: throw new IllegalStateException(); } } static TaskInfo randomTaskInfo() { return randomTaskInfo(randomBoolean()); } static TaskInfo randomTaskInfo(boolean detailed) { TaskId taskId = randomTaskId(); String type = randomAlphaOfLength(5); String action = randomAlphaOfLength(5); Task.Status status = detailed ? randomRawTaskStatus() : null; String description = detailed ? randomAlphaOfLength(5) : null; long startTime = randomLong(); long runningTimeNanos = randomLong(); boolean cancellable = randomBoolean(); boolean cancelled = cancellable == true ? randomBoolean() : false; Long cancellationStartTime = null; if (cancelled) { cancellationStartTime = randomNonNegativeLong(); } TaskId parentTaskId = randomBoolean() ? TaskId.EMPTY_TASK_ID : randomTaskId(); Map<String, String> headers = randomBoolean() ? Collections.emptyMap() : Collections.singletonMap(randomAlphaOfLength(5), randomAlphaOfLength(5)); return new TaskInfo( taskId, type, action, description, status, startTime, runningTimeNanos, cancellable, cancelled, parentTaskId, headers, randomResourceStats(detailed), cancellationStartTime ); } private static TaskId randomTaskId() { return new TaskId(randomAlphaOfLength(5), randomLong()); } private static RawTaskStatus randomRawTaskStatus() { try (XContentBuilder builder = XContentBuilder.builder(Requests.INDEX_CONTENT_TYPE.xContent())) { builder.startObject(); int fields = between(0, 11); for (int f = 0; f < fields; f++) { builder.field(randomAlphaOfLength(5), randomAlphaOfLength(5)); } builder.endObject(); return new RawTaskStatus(BytesReference.bytes(builder)); } catch (IOException e) { throw new IllegalStateException(e); } } public static TaskResourceStats randomResourceStats(boolean detailed) { return detailed ? new TaskResourceStats(new HashMap<>() { { for (int i = 0; i < randomInt(5); i++) { put(randomAlphaOfLength(5), new TaskResourceUsage(randomNonNegativeLong(), randomNonNegativeLong())); } } }, new TaskThreadUsage(randomInt(10), randomInt(10))) : null; } }