gorm.tools.job.SyncJobContext.groovy Maven / Gradle / Ivy
The newest version!
/*
* Copyright 2021 Yak.Works - Licensed under the Apache License, Version 2.0 (the "License")
* You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0
*/
package gorm.tools.job
import java.nio.file.Path
import java.util.concurrent.atomic.AtomicBoolean
import java.util.concurrent.atomic.AtomicInteger
import groovy.json.StreamingJsonBuilder
import groovy.transform.CompileStatic
import groovy.transform.MapConstructor
import groovy.transform.ToString
import groovy.transform.builder.Builder
import groovy.transform.builder.SimpleStrategy
import groovy.util.logging.Slf4j
import org.codehaus.groovy.runtime.StackTraceUtils
import gorm.tools.repository.model.IdGeneratorRepo
import gorm.tools.utils.BenchmarkHelper
import yakworks.api.ApiResults
import yakworks.api.Result
import yakworks.api.ResultUtils
import yakworks.api.problem.Problem
import yakworks.commons.io.IOUtils
import yakworks.commons.lang.Validate
import yakworks.json.groovy.JsonEngine
import yakworks.json.groovy.JsonStreaming
import yakworks.message.spi.MsgService
import yakworks.spring.AppCtx
import static gorm.tools.job.SyncJobArgs.DataFormat
/**
* Holds the basic state and primary action methods for a Bulk job.
* Creates and updates the job status as it progresses and finalizes its results when finished.
*/
@SuppressWarnings('Println')
@Builder(builderStrategy= SimpleStrategy, prefix="")
@MapConstructor
@ToString(includeNames = true, includes = ['ok', 'startTime', 'jobId'])
@Slf4j
@CompileStatic
class SyncJobContext {
/** Thread safe state tracking, any problem will update this to false. */
AtomicBoolean ok = new AtomicBoolean(true)
//reference back to the syncJobService that created this.
SyncJobService syncJobService
/** Arguments for this job.*/
SyncJobArgs args
/** Tracks the start time of the job to use in logging updates*/
Long startTime
/** The master results object */
ApiResults results
/** The problems will be populated if dataFormat="payload", then this will be where they are stored */
List problems = [] as List
/** Payload input data used for job operations */
Object payload
int payloadSize
AtomicInteger processedCount = new AtomicInteger()
AtomicInteger problemCount = new AtomicInteger()
Path dataPath
Closure transformResultsClosure
SyncJobContext() { this([:])}
/** creates a context from the SynJobArgs and assign a back reference to this in SyncJobArgs. */
static SyncJobContext of(SyncJobArgs args){
def sjc = new SyncJobContext(args: args)
args.context = sjc
return sjc
}
/** gets the jobId, stored in args. The job id gets populated once the job is created */
Long getJobId(){ return args.jobId }
/** create a job using the syncJobService.repo.create */
SyncJobContext createJob(){
Validate.notNull(payload)
//get jobId early so it can be used, might not need this anymore
args.jobId = ((IdGeneratorRepo)syncJobService.repo).generateId()
setPayloadSize(payload)
Map data = [
id: args.jobId, source: args.source, sourceId: args.sourceId,
state: SyncJobState.Running, payload: payload
] as Map
if(payload instanceof Collection && payload.size() > 1000) {
args.savePayloadAsFile = true
args.saveDataAsFile = true
}
if(args.savePayload){
if (payload && args.savePayloadAsFile) {
data.payloadId = writePayloadFile(payload as Collection)
}
else {
String res = JsonEngine.toJson(payload)
data.payloadBytes = res.bytes
}
}
//the call to this createJob method is already wrapped in a new trx
def jobEntity = syncJobService.repo.create(data, [flush: true, bindId: true]) as SyncJobEntity
//inititialize the ApiResults to be used in process
results = ApiResults.create()
return this
}
/**
* Update the job results with the current progress info
*
* @param apiResults the ApiResults
* @param startTimeMillis the start time in millis, used to deduce time elapsed
* @param throwEx if false then only does log.error and will not throw on an exception so flow is not disrupted if this flakes out
*/
void updateJobResults(Result apiResults, boolean throwEx = true) {
try {
if(!apiResults.ok) {
ok.set(false)
//if not ok then update the problemCount
int probCnt = (apiResults instanceof ApiResults) ? apiResults.getProblems().size() : 1
problemCount.addAndGet(probCnt)
}
//increment the processedCount
int processedCnt = (apiResults instanceof ApiResults) ? apiResults.list.size() : 1
processedCount.addAndGet(processedCnt)
String message = getJobUpdateMessage(apiResults.ok)
updateJob(apiResults, [id: jobId, ok: ok.get(), message: message])
} catch (e) {
//ok to swallow this excep since we dont want to disrupt the flow
log.error("Unexpected error during updateJobResults", StackTraceUtils.deepSanitize(e))
if(throwEx) throw e
}
}
/**
* Update the total counts for processedCount and problemCount
* and updates message only with the counts, doesn't add or update any results.
* FIXME WIP, needs tests, not used anywhere yet.
*
* @param processedCnt the count to add to the the total processedCount
* @param probCnt the problems to add to the problemCount
*/
void updateMessage(int processedCnt, int probCnt) {
try {
if(processedCnt) processedCount.addAndGet(probCnt)
if(probCnt) problemCount.addAndGet(probCnt)
String message = getJobUpdateMessage(probCnt > 0)
updateJob(null, [id: jobId, ok: ok.get(), message: message])
} catch (e) {
//ok to swallow this excep since we dont want to disrupt the flow, really this should be async
log.error("Unexpected error during updateJobResults", StackTraceUtils.deepSanitize(e))
}
}
/**
* updates job with a result or a problem
*/
void updateWithResult(Result result) {
if(!result.ok) {
ok.set(false)
problemCount.addAndGet(1)
}
updateJob(result, [id: jobId, ok: ok.get()])
}
/**
* Finalize Job by completing data bytes and setting state to Finished.
* @return the finalized SyncJobEntity
*/
SyncJobEntity finishJob() {
Map data = [id: jobId] as Map
if(args.saveDataAsFile){
// if saveDataAsFile then it will have been writing out the data results as it goes
//close out the file
dataPath.withWriterAppend { wr ->
wr.write('\n]\n')
}
data['dataId'] = syncJobService.createAttachment(dataPath, "SyncJobData_${jobId}_.json")
} else {
// if NOT saveDataAsFile then we need to write out the results to dataBytes since results have not been written out yet.
List
© 2015 - 2025 Weber Informatics LLC | Privacy Policy