-
Notifications
You must be signed in to change notification settings - Fork 0
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Added integrations tests for checking workflow creation and update sc… #1
base: composite-workflow
Are you sure you want to change the base?
Changes from all commits
56bba0d
e0af305
feebf0e
22eb900
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,4 +1,3 @@ | ||
package org.opensearch.alerting.model.workflow | ||
|
||
data class WorkflowRunResult { | ||
} | ||
data class WorkflowRunResult(private val someArg: String) |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,160 @@ | ||
/* | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We also need to make changes in TransportDeleteMonitorAction to stop a monitor from being deleted if its a part of an existing composite workflow sequence as a delegate monitor. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Very good point Sashank! |
||
* Copyright OpenSearch Contributors | ||
* SPDX-License-Identifier: Apache-2.0 | ||
*/ | ||
|
||
package org.opensearch.alerting.transport | ||
|
||
import kotlinx.coroutines.CoroutineName | ||
import kotlinx.coroutines.Dispatchers | ||
import kotlinx.coroutines.GlobalScope | ||
import kotlinx.coroutines.launch | ||
import org.apache.logging.log4j.LogManager | ||
import org.opensearch.OpenSearchStatusException | ||
import org.opensearch.action.ActionListener | ||
import org.opensearch.action.ActionRequest | ||
import org.opensearch.action.delete.DeleteRequest | ||
import org.opensearch.action.delete.DeleteResponse | ||
import org.opensearch.action.get.GetRequest | ||
import org.opensearch.action.get.GetResponse | ||
import org.opensearch.action.support.ActionFilters | ||
import org.opensearch.action.support.HandledTransportAction | ||
import org.opensearch.action.support.WriteRequest | ||
import org.opensearch.alerting.opensearchapi.suspendUntil | ||
import org.opensearch.alerting.settings.AlertingSettings | ||
import org.opensearch.alerting.util.AlertingException | ||
import org.opensearch.client.Client | ||
import org.opensearch.cluster.service.ClusterService | ||
import org.opensearch.common.inject.Inject | ||
import org.opensearch.common.settings.Settings | ||
import org.opensearch.common.xcontent.LoggingDeprecationHandler | ||
import org.opensearch.common.xcontent.NamedXContentRegistry | ||
import org.opensearch.common.xcontent.XContentHelper | ||
import org.opensearch.common.xcontent.XContentType | ||
import org.opensearch.commons.alerting.action.AlertingActions | ||
import org.opensearch.commons.alerting.action.DeleteWorkflowRequest | ||
import org.opensearch.commons.alerting.action.DeleteWorkflowResponse | ||
import org.opensearch.commons.alerting.model.ScheduledJob | ||
import org.opensearch.commons.alerting.model.Workflow | ||
import org.opensearch.commons.authuser.User | ||
import org.opensearch.commons.utils.recreateObject | ||
import org.opensearch.index.IndexNotFoundException | ||
import org.opensearch.rest.RestStatus | ||
import org.opensearch.tasks.Task | ||
import org.opensearch.transport.TransportService | ||
|
||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. let's add java docs for classes and comments for methods having complicated logic There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I agree. This class is pretty much obvious and the methods inside it are pretty much descriptive. So probably for indexing the workflow |
||
private val log = LogManager.getLogger(TransportIndexMonitorAction::class.java) | ||
|
||
class TransportDeleteWorkflowAction @Inject constructor( | ||
transportService: TransportService, | ||
val client: Client, | ||
actionFilters: ActionFilters, | ||
val clusterService: ClusterService, | ||
settings: Settings, | ||
val xContentRegistry: NamedXContentRegistry | ||
) : HandledTransportAction<ActionRequest, DeleteWorkflowResponse>( | ||
AlertingActions.DELETE_WORKFLOW_ACTION_NAME, transportService, actionFilters, ::DeleteWorkflowRequest | ||
), | ||
SecureTransportAction { | ||
|
||
@Volatile override var filterByEnabled = AlertingSettings.FILTER_BY_BACKEND_ROLES.get(settings) | ||
|
||
init { | ||
listenFilterBySettingChange(clusterService) | ||
} | ||
|
||
override fun doExecute(task: Task, request: ActionRequest, actionListener: ActionListener<DeleteWorkflowResponse>) { | ||
val transformedRequest = request as? DeleteWorkflowRequest | ||
?: recreateObject(request) { DeleteWorkflowRequest(it) } | ||
|
||
val user = readUserFromThreadContext(client) | ||
val deleteRequest = DeleteRequest(ScheduledJob.SCHEDULED_JOBS_INDEX, transformedRequest.workflowId) | ||
.setRefreshPolicy(WriteRequest.RefreshPolicy.IMMEDIATE) | ||
|
||
if (!validateUserBackendRoles(user, actionListener)) { | ||
return | ||
} | ||
|
||
GlobalScope.launch(Dispatchers.IO + CoroutineName("DeleteWorkflowAction")) { | ||
DeleteWorkflowHandler(client, actionListener, deleteRequest, user, transformedRequest.workflowId).resolveUserAndStart() | ||
} | ||
} | ||
|
||
inner class DeleteWorkflowHandler( | ||
private val client: Client, | ||
private val actionListener: ActionListener<DeleteWorkflowResponse>, | ||
private val deleteRequest: DeleteRequest, | ||
private val user: User?, | ||
private val workflowId: String | ||
) { | ||
suspend fun resolveUserAndStart() { | ||
try { | ||
val workflow = getWorkflow() | ||
|
||
val canDelete = user == null || | ||
!doFilterForUser(user) || | ||
checkUserPermissionsWithResource( | ||
user, | ||
workflow.user, | ||
actionListener, | ||
"workflow", | ||
workflowId | ||
) | ||
|
||
if (canDelete) { | ||
val deleteResponse = deleteWorkflow(workflow) | ||
// TODO - uncomment once the workflow metadata is added | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. can this be uncommented? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Well no - because the workflow metadata are not added still |
||
// deleteMetadata(workflow) | ||
actionListener.onResponse(DeleteWorkflowResponse(deleteResponse.id, deleteResponse.version)) | ||
} else { | ||
actionListener.onFailure( | ||
AlertingException( | ||
"Not allowed to delete this workflow!", | ||
RestStatus.FORBIDDEN, | ||
IllegalStateException() | ||
) | ||
) | ||
} | ||
} catch (t: Exception) { | ||
if (t is IndexNotFoundException) { | ||
actionListener.onFailure( | ||
OpenSearchStatusException( | ||
"Workflow not found.", | ||
RestStatus.NOT_FOUND | ||
) | ||
) | ||
} else { | ||
actionListener.onFailure(AlertingException.wrap(t)) | ||
} | ||
} | ||
} | ||
|
||
private suspend fun getWorkflow(): Workflow { | ||
val getRequest = GetRequest(ScheduledJob.SCHEDULED_JOBS_INDEX, workflowId) | ||
|
||
val getResponse: GetResponse = client.suspendUntil { get(getRequest, it) } | ||
if (getResponse.isExists == false) { | ||
actionListener.onFailure( | ||
AlertingException.wrap( | ||
OpenSearchStatusException("Workflow not found.", RestStatus.NOT_FOUND) | ||
) | ||
) | ||
} | ||
val xcp = XContentHelper.createParser( | ||
xContentRegistry, LoggingDeprecationHandler.INSTANCE, | ||
getResponse.sourceAsBytesRef, XContentType.JSON | ||
) | ||
return ScheduledJob.parse(xcp, getResponse.id, getResponse.version) as Workflow | ||
} | ||
|
||
private suspend fun deleteWorkflow(workflow: Workflow): DeleteResponse { | ||
log.debug("Deleting the workflow with id ${deleteRequest.id()}") | ||
return client.suspendUntil { delete(deleteRequest, it) } | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. how are we handling failure in deletion? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. add debug logs "deleted monitor $monitorId" There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Well we are calling delete(deleteRequest, it) as a lambda. You can see that second parameter of delete function call is it -> which is ActionListener forwarded to a delete function. |
||
} | ||
|
||
private suspend fun deleteMetadata(workflow: Workflow) { | ||
val deleteRequest = DeleteRequest(ScheduledJob.SCHEDULED_JOBS_INDEX, "${workflow.id}-metadata") | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. plz add error handling There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. add debug logs "deleted monitor $monitorMetadata" There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. So just FYI - right now there is no existing workflow metadata. This method is not called at all. Once we introduce workflow metadata I will add error handling. How this sounds? |
||
val deleteResponse: DeleteResponse = client.suspendUntil { delete(deleteRequest, it) } | ||
} | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
we should return list of monitor run results within this class with info from each delegate monitor run.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah - I agree. That's in another PR that contains execution of the workflow changes. Check it out here