All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.vmware.xenon.services.common.ExampleTaskService Maven / Gradle / Ivy

There is a newer version: 1.6.18
Show newest version
/*
 * Copyright (c) 2014-2015 VMware, Inc. All Rights Reserved.
 *
 * 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.vmware.xenon.services.common;

import java.net.URI;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.TimeUnit;
import java.util.function.Consumer;

import com.vmware.xenon.common.FactoryService;
import com.vmware.xenon.common.Operation;
import com.vmware.xenon.common.OperationJoin;
import com.vmware.xenon.common.ServiceDocumentDescription.PropertyUsageOption;
import com.vmware.xenon.common.ServiceHost;
import com.vmware.xenon.common.TaskState;
import com.vmware.xenon.common.TaskState.TaskStage;
import com.vmware.xenon.common.UriUtils;
import com.vmware.xenon.common.Utils;
import com.vmware.xenon.services.common.ExampleService.ExampleServiceState;
import com.vmware.xenon.services.common.QueryTask.Query;

/**
 * This is an example task service. When a new task is posted, it will delete ALL
 * example services. It does this in two steps: querying for the example services
 * and deleting them.
 *
 * Note that this task service will only be authorized to query or delete example
 * services taht are accessible to the user who created the task.
 *
 * This example task service embodies the typical task workflow, which looks like:
 *
 * 1. Client does a POST to the task factory service to create a task service
 * 2. The task service will do some work, almost certainly asychronously. (This
 *    task service does queries and DELETEs to other service.)
 * 3. When the work is completed, the task service will update it's state by
 *    PATCHing itself.
 * 4. When the PATCH is received, if more work is to be done, this will be
 *    repeated as needed.
 *
 * You can think of this as proceeding through a state machine. All tasks have a
 * "stage" {@link TaskStage} that covers just the major parts of the lifecycle,
 * such as CREATED, STARTED, and FINISHED. We'll encode the steps that we do as
 * SubStages. For this task service (and most), those sub stages will be associated
 * with just the STARTED TaskStage.
 *
 */
public class ExampleTaskService
        extends TaskService {

    /**
     * These substages are for tracking the stages unique to our task service. They are only
     * relevant to the STARTED TaskStage. If you create your own task service, these substages
     * will probably be where most of the work happens. See the description above.
     */
    public enum SubStage {
        QUERY_EXAMPLES, DELETE_EXAMPLES
    }

    /** Time in seconds for the task to live */
    private static long DEFAULT_TASK_LIFETIME = 60;

    public static final String FACTORY_LINK = ServiceUriPaths.CORE + "/example-tasks";

    /**
     * Create a default factory service that starts instances of this task service on POST.
     */
    public static FactoryService createFactory() {
        return TaskFactoryService.create(ExampleTaskService.class, ServiceOption.IDEMPOTENT_POST,
                ServiceOption.INSTRUMENTATION);
    }

    public static class ExampleTaskServiceState extends TaskService.TaskServiceState {

        /**
         * Time in seconds before the task expires
         *
         * Technically, this isn't needed: clients can just set documentExpirationTimeMicros.
         * However, this makes tutorials easier: command-line clients can just set this to
         * a constant instead of calculating a future time
         */
        @UsageOption(option = PropertyUsageOption.AUTO_MERGE_IF_NOT_NULL)
        public Long taskLifetime;

        /**
         * The current substage. See {@link SubStage}
         */
        @UsageOption(option = PropertyUsageOption.AUTO_MERGE_IF_NOT_NULL)
        public SubStage subStage;

        /**
         * The query we make to the Query Task service, and the result we
         * get back from it.
         */
        @PropertyOptions(usage = {
                PropertyUsageOption.AUTO_MERGE_IF_NOT_NULL,
                PropertyUsageOption.SERVICE_USE })
        public QueryTask exampleQueryTask;

        /**
         * Optional query clause that will be added, with MUST_OCCUR,
         * to the query used to find example service instances. The default query
         * uses only the document kind to restrict results to example service documents.
         */
        @PropertyOptions(usage = {
                PropertyUsageOption.OPTIONAL })
        public Query customQueryClause;
    }

    public ExampleTaskService() {
        super(ExampleTaskServiceState.class);
        toggleOption(ServiceOption.PERSISTENCE, true);
        toggleOption(ServiceOption.REPLICATION, true);
        toggleOption(ServiceOption.INSTRUMENTATION, true);
        toggleOption(ServiceOption.OWNER_SELECTION, true);
    }

    /**
     * Validate and initialize task state on creation
     */
    @Override
    public void handleStart(Operation post) {
        ExampleTaskServiceState initialState = validateStartPost(post);
        if (initialState == null) {
            return;
        }
        initializeState(initialState, post);

        // complete creation POST
        post.setStatusCode(Operation.STATUS_CODE_ACCEPTED).complete();
        // self patch to start state machine
        sendSelfPatch(initialState, TaskStage.STARTED, null);
    }

    /**
     * Ensure that the input task is valid.
     *
     * Technically we don't need to require a body since there are no parameters. However,
     * non-example tasks will normally have parameters, so this is an example of how they
     * could be validated.
     */
    @Override
    protected ExampleTaskServiceState validateStartPost(Operation taskOperation) {
        ExampleTaskServiceState task = super.validateStartPost(taskOperation);
        if (task == null) {
            return null;
        }

        if (ServiceHost.isServiceCreate(taskOperation)) {
            // apply validation only for the initial creation POST, not restart. Alternatively,
            // this code can exist in the handleCreate method
            if (task.subStage != null) {
                taskOperation.fail(
                        new IllegalArgumentException("Do not specify subStage: internal use only"));
                return null;
            }
            if (task.exampleQueryTask != null) {
                taskOperation.fail(
                        new IllegalArgumentException(
                                "Do not specify exampleQueryTask: internal use only"));
                return null;
            }
        } else {
            // This is a infrastructure start POST, not a client POST.
            // A persisted task might already be finished, but due to node restart
            // ownership changes, the service is started with previously indexed state.
            // We check if the task is in terminal state, so we dont restart it
            if (!TaskState.isInProgress(task.taskInfo)) {
                taskOperation.complete();
                return null;
            }
        }

        if (task.taskLifetime != null && task.taskLifetime <= 0) {
            taskOperation.fail(
                    new IllegalArgumentException("taskLifetime must be positive"));
            return null;
        }

        return task;
    }

    /**
     * Initialize the task
     *
     * We set it to be STARTED: we skip CREATED because we don't need the CREATED state
     * If your task does significant initialization, you may prefer to do it in the
     * CREATED state.
     */
    @Override
    protected void initializeState(ExampleTaskServiceState task, Operation taskOperation) {
        task.subStage = SubStage.QUERY_EXAMPLES;

        if (task.taskLifetime != null) {
            task.documentExpirationTimeMicros = Utils.fromNowMicrosUtc(
                    TimeUnit.SECONDS.toMicros(task.taskLifetime));
        } else if (task.documentExpirationTimeMicros != 0) {
            task.documentExpirationTimeMicros = Utils.fromNowMicrosUtc(
                    TimeUnit.SECONDS.toMicros(DEFAULT_TASK_LIFETIME));
        }

        // Do our task-specific logic... This will allow our ExampleTaskService's "default"
        // expiration to take precedence over the default expiration set by TaskService
        super.initializeState(task, taskOperation);
    }

    /**
     * Handle PATCH
     *
     * All of the work happens through this task service patching itself. It does
     * an operation (for example, querying the example services), and when that completes
     * it updates the task state and progresses to the next step by doing a self PATCH.
     *
     */
    @Override
    public void handlePatch(Operation patch) {
        ExampleTaskServiceState currentTask = getState(patch);
        ExampleTaskServiceState patchBody = getBody(patch);

        if (!validateTransition(patch, currentTask, patchBody)) {
            return;
        }
        updateState(currentTask, patchBody);
        patch.complete();

        switch (patchBody.taskInfo.stage) {
        case CREATED:
            // Won't happen: validateTransition reports error
            break;
        case STARTED:
            handleSubstage(patchBody);
            break;
        case CANCELLED:
            logInfo("Task canceled: not implemented, ignoring");
            break;
        case FINISHED:
            logFine("Task finished successfully");
            break;
        case FAILED:
            logWarning("Task failed: %s", (patchBody.failureMessage == null ? "No reason given"
                    : patchBody.failureMessage));
            break;
        default:
            logWarning("Unexpected stage: %s", patchBody.taskInfo.stage);
            break;
        }
    }

    private void handleSubstage(ExampleTaskServiceState task) {
        switch (task.subStage) {
        case QUERY_EXAMPLES:
            handleQueryExamples(task);
            break;
        case DELETE_EXAMPLES:
            handleDeleteExamples(task);
            break;
        default:
            logWarning("Unexpected sub stage: %s", task.subStage);
            break;
        }
    }

    /**
     * Validate that the PATCH we got requests reasonanble changes to our state
     */
    @Override
    protected boolean validateTransition(Operation patch, ExampleTaskServiceState currentTask,
            ExampleTaskServiceState patchBody) {
        super.validateTransition(patch, currentTask, patchBody);
        if (patchBody.taskInfo.stage == TaskStage.STARTED && patchBody.subStage == null) {
            patch.fail(new IllegalArgumentException("Missing substage"));
            return false;
        }
        if (currentTask.taskInfo != null && currentTask.taskInfo.stage != null) {
            if (currentTask.taskInfo.stage == TaskStage.STARTED
                    && patchBody.taskInfo.stage == TaskStage.STARTED) {
                if (currentTask.subStage.ordinal() > patchBody.subStage.ordinal()) {
                    patch.fail(new IllegalArgumentException("Task substage cannot move backwards"));
                    return false;
                }
            }
        }

        return true;
    }

    /**
     * Handle SubStage QUERY_EXAMPLES.
     *
     * Query the query task service for all example services that we are authorized to access.
     * Authorization is implicit: we do not need to indicate it here.
     */
    private void handleQueryExamples(ExampleTaskServiceState task) {
        // Create a query for "all documents with kind ==
        // com:vmware:xenon:services:common:ExampleService:ExampleServiceState"
        Query.Builder builder = Query.Builder.create()
                .addKindFieldClause(ExampleServiceState.class);

        if (task.customQueryClause != null) {
            // expand query with a specified field value, narrowing down the eligible
            // services
            builder.addClause((task.customQueryClause));
        }

        Query exampleDocumentQuery = builder.build();
        task.exampleQueryTask = QueryTask.Builder.createDirectTask()
                .setQuery(exampleDocumentQuery)
                .build();

        // Send the query to the query task service.
        // When we get a response, advance to the next substage, deleting examples
        // Note that we inherited the authorization context of the incoming patch, so
        // we will only see documents that can be seen by the requesting user.
        // The same is true of our completion: we'll continue to use the same authorization
        // context
        URI queryTaskUri = UriUtils.buildUri(this.getHost(), ServiceUriPaths.CORE_QUERY_TASKS);
        Operation queryRequest = Operation.createPost(queryTaskUri)
                .setBody(task.exampleQueryTask)
                .setCompletion(
                        (op, ex) -> {
                            if (ex != null) {
                                logWarning("Query failed, task will not finish: %s",
                                        ex.getMessage());
                                return;
                            }
                            // We extract the result of the task because DELETE_EXAMPLES will use
                            // the list of documents found
                            task.exampleQueryTask = op.getBody(QueryTask.class);
                            sendSelfPatch(task, TaskStage.STARTED, subStageSetter(SubStage.DELETE_EXAMPLES));
                        });
        sendRequest(queryRequest);
    }

    /**
     * Handle SubStage DELETE_EXAMPLES
     *
     * Delete all of the example service documents that we found in the QUERY_EXAMPLES substage.
     * We do all the DELETEs in parallel by using OperationJoin.
     */
    private void handleDeleteExamples(ExampleTaskServiceState task) {
        if (task.exampleQueryTask.results == null) {
            sendSelfFailurePatch(task, "Query task service returned null results");
            return;
        }

        if (task.exampleQueryTask.results.documentLinks == null) {
            sendSelfFailurePatch(task, "Query task service returned null documentLinks");
            return;
        }
        if (task.exampleQueryTask.results.documentLinks.size() == 0) {
            logFine("No example service documents found, nothing to do");
            sendSelfPatch(task, TaskStage.FINISHED, null);
            return;
        }

        List deleteOperations = new ArrayList<>();
        for (String exampleService : task.exampleQueryTask.results.documentLinks) {
            URI exampleServiceUri = UriUtils.buildUri(this.getHost(), exampleService);
            Operation deleteOp = Operation.createDelete(exampleServiceUri);
            deleteOperations.add(deleteOp);
        }

        // OperationJoin lets us do a set of operations in parallel. If we wanted to,
        // we could specify a batch size to limit the parallelism. We'll receive one
        // completion when all the operations complete.
        OperationJoin operationJoin = OperationJoin.create();
        operationJoin
                .setOperations(deleteOperations)
                .setCompletion((ops, exs) -> {
                    if (exs != null && !exs.isEmpty()) {
                        sendSelfFailurePatch(task, String.format("%d deletes failed", exs.size()));
                        return;
                    } else {
                        sendSelfPatch(task, TaskStage.FINISHED, null);
                    }
                }).sendWith(this);
    }

    /**
     * Helper method that returns a lambda that will set SubStage for us
     * @param subStage the SubStage to use
     *
     * @return lambda helper needed for {@link TaskService#sendSelfPatch(TaskServiceState, TaskStage, Consumer)}
     */
    private Consumer subStageSetter(SubStage subStage) {
        return taskState -> taskState.subStage = subStage;
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy