WorkflowDomainService.cs 200 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124
  1. using Hotline.Caching.Interfaces;
  2. using Hotline.File;
  3. using Hotline.FlowEngine.Definitions;
  4. using Hotline.FlowEngine.Notifications;
  5. using Hotline.FlowEngine.WorkflowModules;
  6. using Hotline.SeedData;
  7. using Hotline.Settings;
  8. using Hotline.Share.Dtos;
  9. using Hotline.Share.Dtos.FlowEngine;
  10. using Hotline.Share.Dtos.FlowEngine.Definition;
  11. using Hotline.Share.Enums.FlowEngine;
  12. using Hotline.Users;
  13. using MapsterMapper;
  14. using Microsoft.Extensions.Logging;
  15. using SqlSugar;
  16. using Hotline.EventBus;
  17. using XF.Domain.Authentications;
  18. using XF.Domain.Dependency;
  19. using XF.Domain.Entities;
  20. using XF.Domain.Exceptions;
  21. using XF.Domain.Repository;
  22. using System.Text;
  23. using FluentValidation;
  24. using Hotline.Configurations;
  25. using Hotline.Share.Dtos.File;
  26. using Hotline.Share.Dtos.FlowEngine.Workflow;
  27. using Hotline.Validators.FlowEngine;
  28. using Microsoft.Extensions.Options;
  29. using Microsoft.AspNetCore.Http;
  30. namespace Hotline.FlowEngine.Workflows
  31. {
  32. public class WorkflowDomainService : IWorkflowDomainService, IScopeDependency
  33. {
  34. private readonly IWorkflowRepository _workflowRepository;
  35. private readonly IRepository<WorkflowStep> _workflowStepRepository;
  36. private readonly IRepository<WorkflowTrace> _workflowTraceRepository;
  37. private readonly IRepository<WorkflowCountersign> _workflowCountersignRepository;
  38. private readonly IMapper _mapper;
  39. private readonly Publisher _publisher;
  40. private readonly ILogger<WorkflowDomainService> _logger;
  41. private readonly IFileRepository _fileRepository;
  42. private readonly IRepository<User> _userRepository;
  43. private readonly ISystemSettingCacheManager _systemSettingCacheManager;
  44. private readonly IWfModuleCacheManager _wfModuleCacheManager;
  45. private readonly ISessionContext _sessionContext;
  46. private readonly IOptionsSnapshot<AppConfiguration> _appOptions;
  47. public WorkflowDomainService(
  48. IWorkflowRepository workflowRepository,
  49. IRepository<WorkflowStep> workflowStepRepository,
  50. IRepository<WorkflowTrace> workflowTraceRepository,
  51. IRepository<WorkflowCountersign> workflowCountersignRepository,
  52. ISystemSettingCacheManager systemSettingCacheManager,
  53. IWfModuleCacheManager wfModuleCacheManager,
  54. ISessionContext sessionContext,
  55. IMapper mapper,
  56. Publisher publisher,
  57. ILogger<WorkflowDomainService> logger,
  58. IOptionsSnapshot<AppConfiguration> appOptions,
  59. IFileRepository fileRepository)
  60. {
  61. _workflowRepository = workflowRepository;
  62. _workflowStepRepository = workflowStepRepository;
  63. _workflowTraceRepository = workflowTraceRepository;
  64. _workflowCountersignRepository = workflowCountersignRepository;
  65. _mapper = mapper;
  66. _publisher = publisher;
  67. _logger = logger;
  68. _fileRepository = fileRepository;
  69. _systemSettingCacheManager = systemSettingCacheManager;
  70. _wfModuleCacheManager = wfModuleCacheManager;
  71. _sessionContext = sessionContext;
  72. _appOptions = appOptions;
  73. }
  74. public async Task<Workflow> CreateWorkflowAsync(WorkflowModule wfModule, string title, string userId,
  75. string orgId, string? externalId = null, CancellationToken cancellationToken = default)
  76. {
  77. var definition = wfModule.Definition;
  78. if (definition is null)
  79. throw new UserFriendlyException("无效流程模板");
  80. var workflow = new Workflow
  81. {
  82. Title = title,
  83. ModuleId = wfModule.Id,
  84. ModuleName = wfModule.Name,
  85. ModuleCode = wfModule.Code,
  86. DefinitionId = definition.Id,
  87. Status = EWorkflowStatus.Runnable,
  88. Steps = new(),
  89. Traces = new(),
  90. WorkflowDefinition = definition,
  91. ExternalId = externalId ?? string.Empty,
  92. // //FlowedOrgIds = new List<string> { orgId },
  93. // FlowedUserIds = new List<string> { userId },
  94. FlowType = definition.FlowType,
  95. };
  96. await _workflowRepository.AddAsync(workflow, cancellationToken);
  97. return workflow;
  98. }
  99. // /// <summary>
  100. // /// 流程开始
  101. // /// </summary>
  102. // public async Task StartAsync(Workflow workflow, WorkflowStep startStep, BasicWorkflowDto dto,
  103. // StepDefine firstStepDefine, bool isNextDynamic, FlowAssignInfo flowAssignInfo,
  104. // ECounterSignType? counterSignType, DateTime? expiredTime,
  105. // CancellationToken cancellationToken)
  106. // {
  107. // if (firstStepDefine.StepType is EStepType.End)
  108. // {
  109. // await _publisher.PublishAsync(
  110. // new StartWorkflowNotify(workflow, dto, flowAssignInfo.FlowAssignType, startStep.WorkflowTrace),
  111. // PublishStrategy.ParallelWhenAll, cancellationToken);
  112. //
  113. // //firstStep是否为end,t: 实际办理节点为startStep, 并且handlerId赋值 f: 实际办理节点为firstStep, handlerId未赋值
  114. // workflow.UpdateActualStepWhenHandle(startStep, _sessionContext.OrgAreaCode,
  115. // _sessionContext.OrgAreaName, _sessionContext.OrgLevel);
  116. //
  117. // workflow.UpdateCurrentStepWhenHandle(startStep, _sessionContext.OrgAreaCode,
  118. // _sessionContext.OrgAreaName, _sessionContext.OrgLevel);
  119. //
  120. // var endTrace = await EndAsync(_sessionContext, workflow, dto, firstStepDefine, startStep, expiredTime,
  121. // cancellationToken);
  122. // return;
  123. // }
  124. //
  125. // var assigner = new UserInfo(
  126. // _sessionContext.UserId,
  127. // _sessionContext.UserName,
  128. // _sessionContext.OrgId,
  129. // _sessionContext.OrgName,
  130. // _sessionContext.OrgIsCenter
  131. // );
  132. //
  133. // //firststeps
  134. // var firstSteps = await CreateNextStepsAsync(workflow, startStep, dto, firstStepDefine, assigner,
  135. // isNextDynamic, expiredTime, dto.IsStartCountersign, cancellationToken: cancellationToken);
  136. //
  137. // await _workflowStepRepository.UpdateAsync(startStep, cancellationToken);
  138. //
  139. // //handle trace
  140. // var trace = await NextTraceAsync(workflow, dto, startStep, cancellationToken);
  141. //
  142. // //指派实际办理节点
  143. // UpdateActualStep(workflow, dto, firstStepDefine, firstSteps);
  144. //
  145. // //更新实际办理节点
  146. // UpdateCurrentStep(workflow, dto, firstStepDefine, firstSteps);
  147. //
  148. // //发起会签时记录顶层会签节点(必须在update currentStep之后)
  149. // if (dto.IsStartCountersign && !workflow.IsInCountersign)
  150. // workflow.StartCountersign(startStep.Id, counterSignType);
  151. //
  152. // await _workflowRepository.UpdateAsync(workflow, cancellationToken);
  153. //
  154. // //publish
  155. // await _publisher.PublishAsync(new StartWorkflowNotify(workflow, dto, flowAssignInfo.FlowAssignType, trace),
  156. // PublishStrategy.ParallelWhenAll, cancellationToken);
  157. // }
  158. /// <summary>
  159. /// new(开启流程并停留在开始节点,开始节点为待办节点,指派给当前操作人)
  160. /// </summary>
  161. public async Task<(Workflow, WorkflowStep)> StartAsync(StartWorkflowDto dto,
  162. string externalId, DateTime? expiredTime = null,
  163. Action<Workflow, WorkflowStep, StepDefine, WorkflowStep>? newStepConfig = null,
  164. CancellationToken cancellationToken = default)
  165. {
  166. // var validator = new StartWorkflowDtoValidator();
  167. // var validResult = await validator.ValidateAsync(dto, cancellationToken);
  168. // if (!validResult.IsValid)
  169. // throw new UserFriendlyException(
  170. // $"非法参数, {string.Join(',', validResult.Errors.Select(d => d.ErrorMessage))}");
  171. var wfModule = await GetWorkflowModuleAsync(dto.DefinitionModuleCode, cancellationToken);
  172. var definition = wfModule.Definition;
  173. if (definition == null)
  174. throw new UserFriendlyException("无效模板编码");
  175. if (definition.Status is not EDefinitionStatus.Enable)
  176. throw new UserFriendlyException("该模板不可用");
  177. //如果发起会签需检查是否支持发起会签
  178. var startStepDefine = definition.FindStartStepDefine();
  179. //下一节点是否为动态节点
  180. var isNextDynamic = startStepDefine.InstanceMode is EInstanceMode.Dynamic &&
  181. !DynamicShouldTerminal(startStepDefine, _sessionContext.OrgLevel);
  182. var firstStepDefine = isNextDynamic
  183. ? startStepDefine
  184. : definition.FindStepDefine(dto.NextStepCode);
  185. if (firstStepDefine is null)
  186. throw new UserFriendlyException("未查询到下一步节点配置");
  187. //1. 如果不是按角色指派,handlers必填 2. 如果按角色指派,handlers可以不选
  188. if (firstStepDefine.HandlerType is not EHandlerType.Role && !dto.NextHandlers.Any())
  189. throw UserFriendlyException.SameMessage("未指派办理人");
  190. if (dto.IsStartCountersign)
  191. {
  192. if (!startStepDefine.CanStartCountersign)
  193. throw new UserFriendlyException("当前节点不支持发起会签");
  194. //if (startStepDefine.HandlerType is EHandlerType.Role)
  195. // throw new UserFriendlyException("当前节点不支持发起会签");
  196. //即使当前节点支持发起会签,但下一节点为信息汇总节点、结束节点时也不可发起会签
  197. if (firstStepDefine.StepType is EStepType.Summary or EStepType.End)
  198. throw new UserFriendlyException("下一节点不允许发起会签");
  199. //下一节点是会签汇总节点也不允许发起会签
  200. if (dto.BackToCountersignEnd)
  201. throw new UserFriendlyException("下一节点不允许发起会签");
  202. }
  203. var workflow = CreateWorkflow(wfModule, dto.Title, externalId);
  204. var defineHandler = startStepDefine.HandlerTypeItems.First();
  205. var assigner = new UserInfo(
  206. _sessionContext.UserId,
  207. _sessionContext.UserName,
  208. _sessionContext.OrgId,
  209. _sessionContext.OrgName,
  210. _sessionContext.OrgIsCenter
  211. );
  212. var handler = new StepAssignInfo
  213. {
  214. Key = _sessionContext.RequiredUserId,
  215. Value = _sessionContext.UserName,
  216. UserId = _sessionContext.UserId,
  217. Username = _sessionContext.UserName,
  218. OrgId = _sessionContext.RequiredOrgId,
  219. OrgName = _sessionContext.OrgName,
  220. RoleId = defineHandler.Key,
  221. RoleName = defineHandler.Value,
  222. FlowAssignType = EFlowAssignType.User
  223. };
  224. var startStep = CreateStartStep(workflow, startStepDefine, dto, assigner, handler, expiredTime, newStepConfig);
  225. if (dto.Files.Any())
  226. startStep.FileJson =
  227. await _fileRepository.AddFileAsync(dto.Files, workflow.ExternalId, startStep.Id, cancellationToken);
  228. await _workflowStepRepository.AddAsync(startStep, cancellationToken);
  229. workflow.Steps.Add(startStep);
  230. //starttrace
  231. // var startTrace = _mapper.Map<WorkflowTrace>(startStep);
  232. // startTrace.StepId = startStep.Id;
  233. // startTrace.TraceType = EWorkflowTraceType.Normal;
  234. // await _workflowTraceRepository.AddAsync(startTrace, cancellationToken);
  235. // workflow.Traces.Add(startTrace);
  236. //starttrace
  237. var startTrace = await CreateTraceAsync(workflow, startStep, cancellationToken: cancellationToken);
  238. startStep.WorkflowTrace = startTrace;
  239. //更新受理人信息
  240. workflow.UpdateAcceptor(
  241. _sessionContext.RequiredUserId,
  242. _sessionContext.UserName,
  243. _sessionContext.StaffNo,
  244. _sessionContext.RequiredOrgId,
  245. _sessionContext.OrgName);
  246. workflow.UpdateActualStepWhenAssign(startStep, handler);
  247. workflow.UpdateCurrentStepWhenAssign(startStep, handler);
  248. await _workflowRepository.AddAsync(workflow, cancellationToken);
  249. // var flowAssignInfo =
  250. // await GetNextStepFlowAssignInfoAsync(workflow, startStep, dto, firstStepDefine, isNextDynamic, cancellationToken);
  251. //publish
  252. await _publisher.PublishAsync(new StartWorkflowNotify(workflow, dto, startTrace),
  253. PublishStrategy.ParallelWhenAll, cancellationToken);
  254. return (workflow, startStep);
  255. }
  256. /// <summary>
  257. /// new
  258. /// workflow, currentStep, nextStepDefine, nextSteps
  259. /// </summary>
  260. public async Task<(Workflow, WorkflowStep, StepDefine, List<WorkflowStep>)> NextAsync(
  261. NextWorkflowDto dto,
  262. EHandleMode handleMode = EHandleMode.Normal,
  263. DateTime? expiredTime = null, bool isAutoFillSummaryOpinion = false,
  264. Action<Workflow, WorkflowStep, StepDefine>? currentStepConfig = null,
  265. Action<Workflow, WorkflowStep, StepDefine, WorkflowStep>? newStepConfig = null,
  266. CancellationToken cancellationToken = default)
  267. {
  268. var workflow = await GetWorkflowAsync(dto.WorkflowId, withDefine: true, withSteps: true,
  269. withTraces: true, withCountersigns: true, cancellationToken: cancellationToken);
  270. CheckWhetherRunnable(workflow.Status);
  271. var currentStep = workflow.Steps.FirstOrDefault(d => d.Id == dto.StepId);
  272. if (currentStep == null)
  273. throw new UserFriendlyException(
  274. $"未找到对应节点, workflowId: {dto.WorkflowId}, stepId: {dto.StepId}", "未找到对应节点");
  275. if (currentStep.Status is EWorkflowStepStatus.Handled)
  276. throw new UserFriendlyException("该状态不支持继续办理");
  277. //todo 校验currentStep 能否由 current 办理
  278. var currentStepDefine = GetStepDefine(workflow.WorkflowDefinition, currentStep.Code);
  279. //下一节点是否为动态节点
  280. var isNextDynamic = (string.IsNullOrEmpty(dto.NextStepCode) || dto.NextStepCode.ToLower() != "end")
  281. && currentStepDefine.InstanceMode is EInstanceMode.Dynamic
  282. && !DynamicShouldTerminal(currentStepDefine, _sessionContext.OrgLevel);
  283. StepDefine nextStepDefine;
  284. if (isNextDynamic
  285. || (workflow.IsInCountersign
  286. && currentStep.IsInCountersign()
  287. && !currentStep.IsTopCountersignEndStep(workflow.TopCountersignStepId))
  288. || dto.IsStartCountersign)
  289. {
  290. //下一步配置为当前节点配置
  291. nextStepDefine = currentStepDefine;
  292. }
  293. else
  294. {
  295. //if dto.NextStepCode is empty, find nextCode, but if not only one throw it
  296. if (string.IsNullOrEmpty(dto.NextStepCode))
  297. {
  298. if (currentStepDefine?.NextSteps?.Count == 0)
  299. throw new UserFriendlyException(
  300. $"未选择下一办理节点且未配置下一节点, workflowId: {workflow.Id}, stepDefineName: {currentStepDefine.Name}",
  301. "未选择下一办理节点");
  302. if (currentStepDefine?.NextSteps?.Count > 1)
  303. throw new UserFriendlyException(
  304. $"未选择下一办理节点且配置有多个节点, workflowId: {workflow.Id}, stepDefineName: {currentStepDefine.Name}",
  305. "未选择下一办理节点且配置有多个节点");
  306. dto.NextStepCode = currentStepDefine.NextSteps.First().Code;
  307. }
  308. //下一步配置为下一步节点配置
  309. nextStepDefine = GetStepDefine(workflow.WorkflowDefinition, dto.NextStepCode);
  310. }
  311. // //需求:按角色选择办理人可以不选,表示该角色下所有人都可以办理,同时依据配置:是否本部门人办理显示待选办理人。角色下只要一人办理即可(即:角色下不发起会签)
  312. // if (!dto.NextHandlers.Any() &&
  313. // nextStepDefine.HandlerType != EHandlerType.Role &&
  314. // nextStepDefine.StepType != EStepType.End &&
  315. // nextStepDefine.StepType != EStepType.Summary
  316. // )
  317. // throw new UserFriendlyException("未指定节点处理者");
  318. if (dto.IsStartCountersign)
  319. {
  320. if (!currentStepDefine.CanStartCountersign)
  321. throw UserFriendlyException.SameMessage("当前节点不支持发起会签");
  322. //即使当前节点支持发起会签,但下一节点为信息汇总节点、结束节点时也不可发起会签
  323. if (nextStepDefine.StepType is EStepType.Summary or EStepType.End)
  324. throw UserFriendlyException.SameMessage("下一汇总节点不允许发起会签");
  325. //下一节点是会签汇总节点也不允许发起会签
  326. if (dto.BackToCountersignEnd)
  327. throw UserFriendlyException.SameMessage("下一会签汇总节点不允许发起会签");
  328. }
  329. // var flowAssignInfo =
  330. // await GetNextStepFlowAssignInfoAsync(workflow, currentStep, dto, nextStepDefine, isNextDynamic, cancellationToken);
  331. dto.FlowAssignType ??= SetNextStepAssignInfo(workflow, currentStep, dto, nextStepDefine, isNextDynamic);
  332. if (!dto.NextHandlers.Any() && nextStepDefine.StepType != EStepType.End)
  333. {
  334. //普通节点往汇总节点办理时可以不选,不选的场景主动去查之前的办理对象
  335. if (nextStepDefine.StepType is EStepType.Summary)
  336. {
  337. var handler = GetSummaryTargetFlowStepHandler(workflow, nextStepDefine.SummaryTargetCode);
  338. dto.NextHandlers = new List<StepAssignInfo> { handler };
  339. }
  340. //需求:按角色选择办理人可以不选,表示该角色下所有人都可以办理,同时依据配置:是否本部门人办理显示待选办理人。角色下只要一人办理即可(即:角色下不发起会签)
  341. else if (nextStepDefine.HandlerType == EHandlerType.Role)
  342. {
  343. var handler = nextStepDefine.HandlerTypeItems.First();
  344. if (nextStepDefine.BusinessType is EBusinessType.DepartmentLeader)
  345. {
  346. dto.NextHandlers.Add(new StepAssignInfo
  347. {
  348. Key = handler.Key,
  349. Value = handler.Value,
  350. RoleId = handler.Key,
  351. RoleName = handler.Value,
  352. OrgId = _sessionContext.RequiredOrgId,
  353. OrgName = _sessionContext.OrgName,
  354. FlowAssignType = EFlowAssignType.OrgAndRole
  355. });
  356. }
  357. else
  358. {
  359. //todo 指派给配置的角色
  360. //dto.FlowAssignType = EFlowAssignType.Role;
  361. dto.NextHandlers.Add(new StepAssignInfo
  362. {
  363. Key = handler.Key,
  364. Value = handler.Value,
  365. RoleId = handler.Key,
  366. RoleName = handler.Value,
  367. FlowAssignType = EFlowAssignType.Role
  368. });
  369. }
  370. }
  371. }
  372. if (!dto.NextHandlers.Any() && nextStepDefine.StepType != EStepType.End)
  373. throw new UserFriendlyException("未指定节点处理者");
  374. #region 办理当前节点
  375. if (dto.Files != null && dto.Files.Any())
  376. currentStep.FileJson = await _fileRepository.AddFileAsync(
  377. dto.Files, workflow.ExternalId, currentStep.Id, cancellationToken);
  378. var counterSignType = GetCounterSignType(dto.IsStartCountersign, currentStep.BusinessType);
  379. var updateSteps = new List<WorkflowStep> { currentStep };
  380. //结束当前会签流程
  381. if (currentStep.IsCountersignEndStep)
  382. {
  383. var countersignStartStep = workflow.Steps.FirstOrDefault(d => d.Id == currentStep.CountersignStartStepId);
  384. if (countersignStartStep is null)
  385. throw new UserFriendlyException(
  386. $"未查询到会签开始step, workflowId: {workflow.Id}, currentStepId: {currentStep.Id}",
  387. "未查询到会签开始节点");
  388. if (countersignStartStep.IsStartCountersign)
  389. {
  390. var currentCountersign =
  391. workflow.Countersigns.FirstOrDefault(d => d.Id == countersignStartStep.StartCountersignId);
  392. if (currentCountersign is null)
  393. throw new UserFriendlyException(
  394. $"未查询到对应会签信息,workflowId:{workflow.Id}, countersignId:{currentStep.CountersignId}",
  395. "无效会签编号");
  396. //结束step会签信息
  397. countersignStartStep.CountersignEnd();
  398. updateSteps.Add(countersignStartStep);
  399. //结束会签
  400. currentCountersign.End(currentStep.Id, currentStep.Code, currentStep.BusinessType,
  401. _sessionContext.RequiredUserId, _sessionContext.UserName,
  402. _sessionContext.RequiredOrgId, _sessionContext.OrgName,
  403. _sessionContext.OrgAreaCode, _sessionContext.OrgAreaName);
  404. await _workflowCountersignRepository.UpdateAsync(currentCountersign, cancellationToken);
  405. }
  406. }
  407. await HandleStepAsync(currentStep, workflow, dto, counterSignType, expiredTime, handleMode, cancellationToken);
  408. currentStep.IsActualHandled = CheckIsActualHandle(workflow, currentStep, nextStepDefine, dto);
  409. _mapper.Map(dto, workflow);
  410. //会签办理节点办理时更新会签members字段
  411. if (currentStep.CountersignPosition is ECountersignPosition.Direct or ECountersignPosition.Indirect)
  412. {
  413. if (!string.IsNullOrEmpty(currentStep.CountersignId))
  414. {
  415. //会签中正常办理节点,更新会签members办理状态
  416. var countersign = workflow.Countersigns.FirstOrDefault(d =>
  417. !d.IsCompleted() && d.Id == currentStep.CountersignId);
  418. if (countersign is not null)
  419. {
  420. //throw new UserFriendlyException(
  421. // $"会签数据异常, workflowId: {currentStep.WorkflowId}, countersignId: {currentStep.CountersignId}",
  422. // "会签数据异常");
  423. countersign.MemberHandled(_sessionContext.UserId, _sessionContext.OrgId);
  424. //update cs
  425. await _workflowCountersignRepository.UpdateNav(countersign)
  426. .Include(d => d.Members)
  427. .ExecuteCommandAsync();
  428. }
  429. }
  430. }
  431. await Task.Run(() => currentStepConfig?.Invoke(workflow, currentStep, nextStepDefine), cancellationToken);
  432. await _workflowStepRepository.UpdateRangeAsync(updateSteps, cancellationToken);
  433. //更新traces
  434. var updateTraces = new List<WorkflowTrace>();
  435. foreach (var updateStep in updateSteps)
  436. {
  437. var updateTrace = workflow.Traces.First(d => d.Id == updateStep.Id);
  438. _mapper.Map(updateStep, updateTrace);
  439. updateTraces.Add(updateTrace);
  440. }
  441. await _workflowTraceRepository.UpdateRangeAsync(updateTraces, cancellationToken);
  442. if (workflow.ActualHandleStepId == currentStep.Id)
  443. {
  444. //更新实际办理节点信息
  445. workflow.UpdateActualStepWhenHandle(currentStep, _sessionContext.OrgAreaCode,
  446. _sessionContext.OrgAreaName, _sessionContext.OrgLevel);
  447. }
  448. if (workflow.CurrentStepId == currentStep.Id)
  449. {
  450. workflow.UpdateCurrentStepWhenHandle(currentStep, _sessionContext.OrgAreaCode,
  451. _sessionContext.OrgAreaName, _sessionContext.OrgLevel);
  452. }
  453. //var trace = await NextTraceAsync(workflow, dto, currentStep, cancellationToken);
  454. #endregion
  455. #region 处理流程
  456. //检查会签是否结束,并更新当前会签节点字段
  457. var isCountersignOver = false;
  458. if (workflow.IsInCountersign && currentStep.IsCountersignEndStep)
  459. {
  460. isCountersignOver = workflow.CheckIfCountersignOver();
  461. if (isCountersignOver)
  462. workflow.EndCountersign();
  463. }
  464. //检查是否流转到流程终点
  465. if (nextStepDefine.StepType is EStepType.End)
  466. {
  467. if (string.IsNullOrEmpty(workflow.ActualHandlerId)
  468. || string.IsNullOrEmpty(workflow.ActualHandleOrgCode)) //开始流程直接归档
  469. {
  470. //更新实际办理节点信息
  471. workflow.UpdateActualStepWhenHandle(currentStep, _sessionContext.OrgAreaCode,
  472. _sessionContext.OrgAreaName, _sessionContext.OrgLevel);
  473. workflow.ActualHandleStepAcceptTime = currentStep.AcceptTime;
  474. workflow.ActualHandleTime = currentStep.HandleTime;
  475. }
  476. var (endStep, _) = await EndAsync(_sessionContext, workflow, dto, nextStepDefine, currentStep, expiredTime, cancellationToken);
  477. return (workflow, currentStep, nextStepDefine, new List<WorkflowStep> { endStep });
  478. }
  479. var isStartCountersign = currentStep.CountersignPosition switch
  480. {
  481. ECountersignPosition.None => dto.IsStartCountersign,
  482. ECountersignPosition.Direct => !dto.BackToCountersignEnd,
  483. ECountersignPosition.Indirect => !dto.BackToCountersignEnd,
  484. ECountersignPosition.End => !dto.BackToCountersignEnd && (workflow.IsInCountersign || dto.IsStartCountersign),
  485. _ => throw new ArgumentOutOfRangeException()
  486. };
  487. var currentTrace = workflow.Traces.First(d => d.Id == currentStep.Id);
  488. //创建会签数据
  489. if (isStartCountersign)
  490. {
  491. var exists = workflow.Countersigns.Any(d =>
  492. !d.IsCompleted() && !string.IsNullOrEmpty(_sessionContext.UserId) && d.StarterId == _sessionContext.UserId);
  493. if (exists)
  494. throw new UserFriendlyException($"该用户在当前流程存在未结束会签, workflowId: {workflow.Id}, userId: {_sessionContext.UserId}");
  495. var countersign = await StartCountersignAsync(_sessionContext, workflow, currentStep, dto,
  496. counterSignType, expiredTime, cancellationToken);
  497. currentTrace.StartCountersign(countersign.Id);
  498. await _workflowStepRepository.UpdateAsync(currentStep, cancellationToken);
  499. await _workflowTraceRepository.UpdateAsync(currentTrace, cancellationToken);
  500. }
  501. //发起会签时记录顶层会签节点
  502. if (dto.IsStartCountersign && !workflow.IsInCountersign)
  503. workflow.StartCountersign(currentStep.Id, counterSignType);
  504. var assigner = new UserInfo(
  505. _sessionContext.UserId,
  506. _sessionContext.UserName,
  507. _sessionContext.OrgId,
  508. _sessionContext.OrgName,
  509. _sessionContext.OrgIsCenter
  510. );
  511. //创建下一/N个节点(会签汇总节点:会签未全部办理时不创建,最后一个会签办理节点创建会签汇总节点)
  512. var nextSteps = await CreateNextStepsAsync(workflow, currentStep, dto,
  513. nextStepDefine, assigner, isNextDynamic, expiredTime, dto.IsStartCountersign,
  514. isAutoFillSummaryOpinion, newStepConfig, cancellationToken);
  515. // //更新办理对象(nextSteps无元素表示当前节点为会签办理节点且当前会签没有全部办理完成)
  516. // workflow.UpdateHandlers(current.RequiredUserId, current.RequiredOrgId,
  517. // flowAssignInfo.FlowAssignType, flowAssignInfo.HandlerObjects, nextSteps.Any());
  518. //指派实际办理节点
  519. UpdateActualStep(workflow, dto, nextStepDefine, nextSteps);
  520. //更新实际办理节点
  521. UpdateCurrentStep(workflow, dto, nextStepDefine, nextSteps);
  522. //更新会签实际办理对象信息
  523. if (currentStep.IsActualHandled)
  524. workflow.AddCsActualHandler(_sessionContext.UserId, _sessionContext.OrgId);
  525. await _workflowRepository.UpdateAsync(workflow, cancellationToken);
  526. #endregion
  527. #region 流转记录
  528. //var trace = await NextTraceAsync(workflow, dto, currentStep, cancellationToken);
  529. #endregion
  530. await _publisher.PublishAsync(
  531. new NextStepNotify(workflow, dto, currentTrace, nextStepDefine,
  532. _sessionContext.OrgId, expiredTime.HasValue), PublishStrategy.ParallelWhenAll,
  533. cancellationToken);
  534. return (workflow, currentStep, nextStepDefine, nextSteps);
  535. }
  536. /// <summary>
  537. /// 开启流程并办理至第一个节点
  538. /// </summary>
  539. public async Task<List<WorkflowStep>> StartToFirstStepAsync(StartWorkflowDto dto,
  540. string externalId, DateTime? expiredTime = null,
  541. Action<Workflow, WorkflowStep, StepDefine, WorkflowStep>? newStepConfig = null,
  542. CancellationToken cancellationToken = default)
  543. {
  544. var (workflow, startStep) = await StartAsync(dto, externalId, expiredTime, newStepConfig, cancellationToken);
  545. var nextDto = _mapper.Map<NextWorkflowDto>(dto);
  546. nextDto.WorkflowId = workflow.Id;
  547. nextDto.StepId = startStep.Id;
  548. var (_, _, _, nextSteps) =
  549. await NextAsync(nextDto,
  550. expiredTime: expiredTime,
  551. isAutoFillSummaryOpinion: true,
  552. newStepConfig: newStepConfig, cancellationToken: cancellationToken);
  553. return nextSteps;
  554. }
  555. public async Task<Workflow> GetWorkflowAsync(string workflowId,
  556. bool withDefine = false, bool withSteps = false,
  557. bool withTraces = false, bool withTracesTree = false,
  558. bool withSupplements = false, bool withCountersigns = false,
  559. CancellationToken cancellationToken = default)
  560. {
  561. if (withTraces && withTracesTree)
  562. throw new UserFriendlyException("traces只能在集合与树形集合结构中二选一");
  563. var query = _workflowRepository.Queryable().Where(d => d.Id == workflowId);
  564. if (withDefine)
  565. query = query.Includes(d => d.WorkflowDefinition);
  566. // if (withSupplements)
  567. // query = query.Includes(d => d.Supplements);
  568. //if (withAssigns)
  569. // query = query.Includes(d => d.Assigns);
  570. if (withCountersigns)
  571. query = query.Includes(d => d.Countersigns, x => x.Members);
  572. if (withSteps)
  573. query = query.Includes(d => d.Steps);
  574. if (withTraces)
  575. query = query.Includes(d => d.Traces);
  576. var workflow = await query.FirstAsync(cancellationToken);
  577. if (workflow is null)
  578. throw new UserFriendlyException("无效workflowId");
  579. if (withTracesTree)
  580. {
  581. workflow.Traces = await _workflowTraceRepository.Queryable()
  582. .Where(d => d.WorkflowId == workflow.Id)
  583. .OrderBy(d => d.CreationTime)
  584. .ToTreeAsync(d => d.Traces, d => d.ParentId, null);
  585. }
  586. return workflow;
  587. }
  588. /// <summary>
  589. /// 查询用户对于当前流程权限
  590. /// </summary>
  591. public async Task<(Workflow Workflow, string? CountersignId, bool CanHandle, bool CanPrevious, WorkflowTrace? Trace)>
  592. GetWorkflowHandlePermissionAsync(
  593. string workflowId, string userId, string orgId, string[] roleIds,
  594. CancellationToken cancellationToken = default)
  595. {
  596. var workflow = await GetWorkflowAsync(workflowId,
  597. withSteps: true, withTraces: true, withCountersigns: true,
  598. cancellationToken: cancellationToken);
  599. var canHandle = workflow.IsCanHandle(userId, orgId, roleIds);
  600. var canPrevious = false;
  601. if (canHandle)
  602. {
  603. var currentStep = FindCurrentStepWaitForHandle(workflow, userId, orgId, roleIds);
  604. if (currentStep.Status is not EWorkflowStepStatus.Handled)
  605. {
  606. canPrevious = !(currentStep.IsInCountersign() &&
  607. !currentStep.IsTopCountersignEndStep(workflow.TopCountersignStepId));
  608. }
  609. }
  610. var unhandlePreviousTrace = workflow.Traces.FirstOrDefault(d =>
  611. d.Status is not EWorkflowStepStatus.Handled
  612. );
  613. var unCompletedCountersign = workflow.Countersigns
  614. .FirstOrDefault(d => !d.IsCompleted() && d.StarterOrgId == orgId);
  615. if (unCompletedCountersign is null)
  616. return (workflow, null, canHandle, canPrevious, unhandlePreviousTrace);
  617. //var existCountersignEndStep = workflow.Steps.Exists(d =>
  618. // d.IsCountersignEndStep && d.CountersignStartStepId == unCompletedCountersign.StartStepId);
  619. //return (workflow, existCountersignEndStep ? null : unCompletedCountersign.Id, canPrevious);
  620. return (workflow, unCompletedCountersign?.Id ?? null, canHandle, canPrevious, unhandlePreviousTrace);
  621. }
  622. /// <summary>
  623. /// 受理(接办)
  624. /// </summary>
  625. public async Task<DateTime?> AcceptAsync(Workflow workflow,
  626. string userId, string? userName,
  627. string orgId, string? orgName,
  628. string? orgAreaCode, string? orgAreaName,
  629. CancellationToken cancellationToken)
  630. {
  631. if (!workflow.IsCanHandle(_sessionContext.RequiredUserId, _sessionContext.RequiredOrgId,
  632. _sessionContext.Roles)) return null;
  633. //工单完成以后查看的场景
  634. if (workflow.Status != EWorkflowStatus.Runnable) return null;
  635. var currentStep = GetUnHandleStep(workflow.Steps, _sessionContext.RequiredOrgId,
  636. _sessionContext.RequiredUserId, _sessionContext.Roles);
  637. if (currentStep.Status is not EWorkflowStepStatus.WaitForAccept) return null;
  638. //if (currentStep.Handlers.All(d => d.Key != orgId && d.Key != userId)) return null;
  639. if (currentStep.StepType is EStepType.End)
  640. throw new UserFriendlyException("当前流程已流转到最终步骤");
  641. currentStep.Accept(userId, userName,
  642. orgId, orgName,
  643. orgAreaCode, orgAreaName);
  644. var trace = workflow.Traces.First(d => d.Id == currentStep.Id);
  645. _mapper.Map(currentStep, trace);
  646. currentStep.WorkflowTrace = trace;
  647. await _workflowStepRepository.UpdateNav(currentStep)
  648. .Include(d => d.WorkflowTrace)
  649. .ExecuteCommandAsync();
  650. workflow.ActualHandleStepAcceptTime = currentStep.AcceptTime;
  651. workflow.CurrentStepAcceptTime = currentStep.AcceptTime;
  652. await _workflowRepository.Updateable(workflow).ExecuteCommandAsync(cancellationToken);
  653. return workflow.ActualHandleStepAcceptTime;
  654. }
  655. ///// <summary>
  656. ///// 退回(返回前一节点)
  657. ///// </summary>
  658. ///// <returns></returns>
  659. //public async Task<EFlowDirection> PreviousAsync(Workflow workflow, PreviousWorkflowDto dto,
  660. // string applicantId, string applicantName,
  661. // string applicantOrgId, string applicantOrgName,
  662. // string applicantOrgAreaCode, string applicantOrgAreaName,
  663. // bool applicantIsCenter, string[] applicantRoleIds,
  664. // CancellationToken cancellationToken)
  665. //{
  666. // //ValidatePermission(workflow, operater.OrgId, operater.Id);
  667. // var (currentStep, prevStep, countersignStartStep) =
  668. // GetPreviousStep(workflow, applicantId, applicantOrgId, applicantRoleIds);
  669. // //保存附件
  670. // if (dto.Files.Any())
  671. // currentStep.FileJson = await _fileRepository.AddFileAsync(
  672. // dto.Files, workflow.ExternalId, currentStep.Id, cancellationToken);
  673. // // add prev current to remove list
  674. // var removeSteps = new List<WorkflowStep> { currentStep, prevStep };
  675. // if (countersignStartStep is not null)
  676. // {
  677. // //add cs steps to remove list
  678. // SearchCountersignSteps(countersignStartStep, workflow.Steps, ref removeSteps);
  679. // //end cs
  680. // var currentCountersign =
  681. // workflow.Countersigns.FirstOrDefault(d => d.Id == countersignStartStep.StartCountersignId);
  682. // if (currentCountersign is null)
  683. // throw new UserFriendlyException(
  684. // $"未查询到对应会签信息,workflowId:{workflow.Id}, countersignId:{currentStep.CountersignId}",
  685. // "无效会签编号");
  686. // //结束step会签信息
  687. // countersignStartStep.CountersignEnd();
  688. // await _workflowStepRepository.UpdateAsync(countersignStartStep, cancellationToken);
  689. // //updateSteps.Add(countersignStartStep);
  690. // //结束会签
  691. // //currentCountersign.End(currentStep.Id, currentStep.Code, currentStep.BusinessType,
  692. // // current.RequiredUserId, current.UserName,
  693. // // current.RequiredOrgId, current.OrgName,
  694. // // current.OrgAreaCode, current.OrgAreaName);
  695. // currentCountersign.End(currentStep.Id, currentStep.Code, currentStep.BusinessType,
  696. // applicantId, applicantName,
  697. // applicantOrgId, applicantOrgName,
  698. // applicantOrgAreaCode, applicantOrgAreaName);
  699. // await _workflowCountersignRepository.UpdateAsync(currentCountersign, cancellationToken);
  700. // //update workflow cs status
  701. // if (workflow.CheckIfCountersignOver())
  702. // workflow.EndCountersign();
  703. // }
  704. // //update trace
  705. // //var trace = await PreviousTraceAsync(workflow.Id, dto, currentStep,
  706. // // applicantId, applicantName,
  707. // // applicantOrgId, applicantOrgName,
  708. // // applicantOrgAreaCode, applicantOrgAreaName,
  709. // // applicantIsCenter, cancellationToken);
  710. // var trace = workflow.Traces.First(t => t.StepId == currentStep.Id);
  711. // // _mapper.Map(dto, trace);
  712. // trace.FileJson = currentStep.FileJson;
  713. // trace.IsSms = dto.AcceptSms;
  714. // trace.Opinion = dto.Opinion;
  715. // //HandleTrace(trace, dto.Opinion, current);
  716. // trace.Handle(applicantId, applicantName,
  717. // applicantOrgId, applicantOrgName,
  718. // applicantOrgAreaCode, applicantOrgAreaName,
  719. // applicantIsCenter, EHandleMode.Previous, dto.Opinion);
  720. // //await _workflowTraceRepository.UpdateAsync(trace, cancellationToken);
  721. // //如果有传入期满时间 新节点为传入的期满时间
  722. // if (dto.ExpiredTime.HasValue)
  723. // prevStep.StepExpiredTime = dto.ExpiredTime;
  724. // if (workflow.FlowType == EFlowType.Handle)
  725. // {
  726. // prevStep.FlowAssignType = prevStep.BusinessType is EBusinessType.Seat ? EFlowAssignType.Role :
  727. // prevStep.BusinessType is EBusinessType.Send ? EFlowAssignType.User : EFlowAssignType.Org;
  728. // }
  729. // //甄别退回到最开始节点到部门
  730. // if (workflow.FlowType == EFlowType.Review && workflow.ModuleCode == WorkflowModuleConsts.OrderScreen)
  731. // {
  732. // prevStep.FlowAssignType = prevStep.StepType == EStepType.Start ? EFlowAssignType.Org : prevStep.FlowAssignType;
  733. // }
  734. // //复制上一个节点为待接办
  735. // // var newPrevStep =
  736. // // await DuplicateStepWithTraceAsync(workflow, prevStep, EWorkflowTraceType.Previous, cancellationToken);
  737. // var newPrevStep = DuplicateStep(prevStep, EWorkflowTraceType.Previous, dto.ExpiredTime);
  738. // //退给派单组节点,需按照平均分配原则派给一个派单员 禅道299 TODO
  739. // if (dto.Handler != null) //todo 改为按策略判断
  740. // {
  741. // var handle = dto.Handler;
  742. // newPrevStep.Assign(handle.UserId, handle.Username, handle.OrgId, handle.OrgName, handle.RoleId, handle.RoleName);
  743. // }
  744. // await _workflowStepRepository.AddAsync(newPrevStep, cancellationToken);
  745. // await CreateTraceAsync(workflow, newPrevStep, EWorkflowTraceType.Previous, cancellationToken);
  746. // //remove workflow.steps
  747. // await _workflowStepRepository.RemoveRangeAsync(removeSteps, cancellationToken);
  748. // var stepIds = removeSteps.Select(d => d.Id).ToList();
  749. // var updateTraces = workflow.Traces.Where(d => stepIds.Contains(d.StepId)).ToList();
  750. // await UpdateTracesStateAsync(updateTraces, EWorkflowTraceState.StepRemoveByPrevious, cancellationToken);
  751. // if (workflow.Status is EWorkflowStatus.Completed)
  752. // workflow.SetStatusRunnable();
  753. // //更新实际办理节点信息
  754. // workflow.UpdateActualStepWhenAssign(newPrevStep, new FlowStepHandler
  755. // {
  756. // UserId = newPrevStep.HandlerId,
  757. // Username = newPrevStep.HandlerName,
  758. // OrgId = newPrevStep.HandlerOrgId,
  759. // OrgName = newPrevStep.HandlerOrgName,
  760. // });
  761. // workflow.UpdateCurrentStepWhenAssign(newPrevStep, new FlowStepHandler
  762. // {
  763. // UserId = newPrevStep.HandlerId,
  764. // Username = newPrevStep.HandlerName,
  765. // OrgId = newPrevStep.HandlerOrgId,
  766. // OrgName = newPrevStep.HandlerOrgName,
  767. // });
  768. // // //更新流程可办理对象
  769. // // workflow.UpdatePreviousHandlers(applicantId, applicantOrgId, prevStep);
  770. // //orgToCenter会触发重新计算期满时间,1.无需审核按当前时间进行计算 2.需审核按审核通过时间计算
  771. // var isOrgToCenter = prevStep.BusinessType is EBusinessType.Send && prevStep.IsOrigin;
  772. // await _workflowRepository.UpdateAsync(workflow, cancellationToken);
  773. // await _publisher.PublishAsync(new PreviousNotify(workflow, newPrevStep, dto, isOrgToCenter),
  774. // PublishStrategy.ParallelWhenAll, cancellationToken);
  775. // return GetFlowDirection(currentStep.BusinessType, prevStep.BusinessType);
  776. //}
  777. /// <summary>
  778. /// 退回(new)
  779. /// </summary>
  780. public async Task<(Workflow workflow, WorkflowStep currentStep, StepDefine prevDefine,
  781. WorkflowStep prevStep, WorkflowStep newStep, EFlowDirection flowDirection)>
  782. PreviousAsync(Workflow workflow, PreviousWorkflowDto dto, OperatorInfo operatorInfo,
  783. EHandleMode handleMode = EHandleMode.Previous,
  784. Action<Workflow, WorkflowStep, StepDefine, WorkflowStep, WorkflowStep>? newStepConfig = null,
  785. CancellationToken cancellationToken = default)
  786. {
  787. //ValidatePermission(workflow, operater.OrgId, operater.Id);
  788. if (string.IsNullOrEmpty(operatorInfo.UserId)
  789. && string.IsNullOrEmpty(operatorInfo.OrgId)
  790. && !operatorInfo.Roles.Any())
  791. throw new UserFriendlyException("无效当前操作人信息");
  792. var (currentStep, prevStep, countersignStartStep) =
  793. GetPreviousStep(workflow, operatorInfo.UserId, operatorInfo.OrgId, operatorInfo.Roles);
  794. //保存附件
  795. if (dto.Files.Any())
  796. currentStep.FileJson = await _fileRepository.AddFileAsync(dto.Files, workflow.ExternalId, currentStep.Id, cancellationToken);
  797. // add prev current to remove list
  798. var removeSteps = new List<WorkflowStep> { currentStep, prevStep };
  799. if (countersignStartStep is not null)
  800. {
  801. //add cs steps to remove list
  802. SearchCountersignSteps(countersignStartStep, workflow.Steps, ref removeSteps);
  803. //end cs
  804. var currentCountersign =
  805. workflow.Countersigns.FirstOrDefault(d => d.Id == countersignStartStep.StartCountersignId);
  806. if (currentCountersign is null)
  807. throw new UserFriendlyException(
  808. $"未查询到对应会签信息,workflowId:{workflow.Id}, countersignId:{currentStep.CountersignId}",
  809. "无效会签编号");
  810. //结束step会签信息
  811. countersignStartStep.CountersignEnd();
  812. await _workflowStepRepository.UpdateAsync(countersignStartStep, cancellationToken);
  813. //updateSteps.Add(countersignStartStep);
  814. //结束会签
  815. currentCountersign.End(currentStep.Id, currentStep.Code, currentStep.BusinessType,
  816. operatorInfo.UserId, operatorInfo.UserName,
  817. operatorInfo.OrgId, operatorInfo.OrgName,
  818. operatorInfo.OrgAreaCode, operatorInfo.OrgAreaName);
  819. await _workflowCountersignRepository.UpdateAsync(currentCountersign, cancellationToken);
  820. //update workflow cs status
  821. if (workflow.CheckIfCountersignOver())
  822. workflow.EndCountersign();
  823. }
  824. var trace = workflow.Traces.First(t => t.StepId == currentStep.Id);
  825. // _mapper.Map(dto, trace);
  826. trace.FileJson = currentStep.FileJson;
  827. trace.IsSms = dto.AcceptSms;
  828. trace.Opinion = dto.Opinion;
  829. //HandleTrace(trace, dto.Opinion, current);
  830. trace.Handle(operatorInfo.UserId, operatorInfo.UserName,
  831. operatorInfo.OrgId, operatorInfo.OrgName,
  832. operatorInfo.OrgAreaCode, operatorInfo.OrgAreaName,
  833. operatorInfo.OrgIsCenter, handleMode, dto.Opinion);
  834. //如果有传入期满时间 新节点为传入的期满时间
  835. if (dto.ExpiredTime.HasValue)
  836. prevStep.StepExpiredTime = dto.ExpiredTime;
  837. //if (workflow.FlowType == EFlowType.Handle) //该逻辑需放在退回操作前依据业务判断
  838. //{
  839. // prevStep.FlowAssignType = prevStep.BusinessType is EBusinessType.Seat ? EFlowAssignType.Role :
  840. // prevStep.BusinessType is EBusinessType.Send ? EFlowAssignType.User : EFlowAssignType.Org;
  841. //}
  842. // dto.ReverseFlowStepAssignInfo ??= new ReverseFlowStepAssignInfo(EReverseFlowStepCreationPolicy.OriginStepUser);
  843. // //todo
  844. // var stepAssignInfo = GetStepAssignInfo(dto.ReverseFlowStepAssignInfo, prevStep, prevStepDefine);
  845. var prevStepDefine = workflow.WorkflowDefinition.FindStepDefine(prevStep.Code);
  846. //复制上一个节点为待接办
  847. var newPrevStep = DuplicateStep(workflow, prevStepDefine, prevStep, EWorkflowTraceType.Previous, dto.ExpiredTime);
  848. newPrevStep.Assign(prevStep, EFlowAssignType.User);
  849. //甄别退回到最开始节点到部门 todo 重构放在调用处判断
  850. if (workflow.FlowType == EFlowType.Review && workflow.ModuleCode == WorkflowModuleConsts.OrderScreen)
  851. {
  852. newPrevStep.FlowAssignType = newPrevStep.StepType == EStepType.Start ? EFlowAssignType.Org : prevStep.FlowAssignType;
  853. }
  854. //退给派单组节点,需按照平均分配原则派给一个派单员 禅道299 TODO
  855. if (dto.Handler != null) //todo 改为按策略判断
  856. {
  857. var handle = dto.Handler;
  858. newPrevStep.Assign(handle.UserId, handle.Username, handle.OrgId, handle.OrgName, handle.RoleId, handle.RoleName);
  859. }
  860. //await Task.Run(() => newStepConfig?.Invoke(workflow, currentStep, prevStepDefine, prevStep, newPrevStep), cancellationToken);
  861. newStepConfig?.Invoke(workflow, currentStep, prevStepDefine, prevStep, newPrevStep);
  862. await _workflowStepRepository.AddAsync(newPrevStep, cancellationToken);
  863. await CreateTraceAsync(workflow, newPrevStep, EWorkflowTraceType.Previous, cancellationToken);
  864. //remove workflow.steps
  865. await _workflowStepRepository.RemoveRangeAsync(removeSteps, cancellationToken);
  866. var stepIds = removeSteps.Select(d => d.Id).ToList();
  867. var updateTraces = workflow.Traces.Where(d => stepIds.Contains(d.StepId)).ToList();
  868. await UpdateTracesStateAsync(updateTraces, EWorkflowTraceState.StepRemoveByPrevious, cancellationToken);
  869. if (workflow.Status is EWorkflowStatus.Completed)
  870. workflow.SetStatusRunnable();
  871. //更新实际办理节点信息
  872. workflow.UpdateActualStepWhenAssign(newPrevStep, new FlowStepHandler
  873. {
  874. UserId = newPrevStep.HandlerId,
  875. Username = newPrevStep.HandlerName,
  876. OrgId = newPrevStep.HandlerOrgId,
  877. OrgName = newPrevStep.HandlerOrgName,
  878. });
  879. workflow.UpdateCurrentStepWhenAssign(newPrevStep, new FlowStepHandler
  880. {
  881. UserId = newPrevStep.HandlerId,
  882. Username = newPrevStep.HandlerName,
  883. OrgId = newPrevStep.HandlerOrgId,
  884. OrgName = newPrevStep.HandlerOrgName,
  885. });
  886. // //更新流程可办理对象
  887. // workflow.UpdatePreviousHandlers(applicantId, applicantOrgId, prevStep);
  888. //orgToCenter会触发重新计算期满时间,1.无需审核按当前时间进行计算 2.需审核按审核通过时间计算
  889. var isOrgToCenter = prevStep.BusinessType is EBusinessType.Send && prevStep.IsOrigin;
  890. await _workflowRepository.UpdateAsync(workflow, cancellationToken);
  891. await _publisher.PublishAsync(new PreviousNotify(workflow, newPrevStep, dto, isOrgToCenter),
  892. PublishStrategy.ParallelWhenAll, cancellationToken);
  893. var flowDirection = GetFlowDirection(currentStep.BusinessType, prevStep.BusinessType);
  894. return (workflow, currentStep, prevStepDefine, prevStep, newPrevStep, flowDirection);
  895. }
  896. private async Task UpdateTracesStateAsync(List<WorkflowTrace> traces, EWorkflowTraceState traceState,
  897. CancellationToken cancellationToken)
  898. {
  899. foreach (var trace in traces)
  900. {
  901. trace.TraceState = traceState;
  902. }
  903. await _workflowTraceRepository.UpdateRangeAsync(traces, cancellationToken);
  904. }
  905. /// <summary>
  906. /// 查询退回节点信息
  907. /// </summary>
  908. public (WorkflowStep currentStep, WorkflowStep prevStep, WorkflowStep? countersignStartStep) GetPreviousStep(
  909. Workflow workflow, string operaterId, string operaterOrgId, string[] roleIds)
  910. {
  911. var currentStep = GetUnHandleStep(workflow.Steps, operaterOrgId, operaterId, roleIds);
  912. var isCurrentTopCountersignEndStep = workflow.IsInCountersign &&
  913. currentStep.IsTopCountersignEndStep(workflow.TopCountersignStepId);
  914. if (currentStep.IsInCountersign() && !isCurrentTopCountersignEndStep)
  915. throw UserFriendlyException.SameMessage("会签节点不支持退回");
  916. if (workflow.FlowType is EFlowType.Review && currentStep.StepType is EStepType.Start &&
  917. currentStep.IsOrigin)
  918. throw UserFriendlyException.SameMessage("当前流程已退回到开始节点");
  919. //当退回操作遇到会签时,删除所有会签节点直达topCsStep
  920. //find prevStep, update handler
  921. WorkflowStep? prevStep, countersignStartStep = null;
  922. if (isCurrentTopCountersignEndStep)
  923. {
  924. //prev is topstart's prev
  925. countersignStartStep = workflow.Steps.FirstOrDefault(d => d.Id == currentStep.CountersignStartStepId);
  926. if (countersignStartStep is null)
  927. throw new UserFriendlyException("未查询到对应会签开始节点");
  928. prevStep = workflow.Steps.FirstOrDefault(d => d.Id == countersignStartStep.PrevStepId);
  929. }
  930. else
  931. {
  932. prevStep = workflow.Steps.FirstOrDefault(d => d.Id == currentStep.PrevStepId);
  933. }
  934. if (prevStep == null)
  935. throw UserFriendlyException.SameMessage("未查询到前一节点");
  936. while (prevStep.IsCountersignEndStep)
  937. {
  938. countersignStartStep = workflow.Steps.FirstOrDefault(d => d.Id == prevStep.CountersignStartStepId);
  939. prevStep = countersignStartStep ?? throw new UserFriendlyException("未查询到对应会签开始节点");
  940. if (prevStep == null)
  941. throw UserFriendlyException.SameMessage("未查询到前一节点");
  942. }
  943. return (currentStep, prevStep, countersignStartStep);
  944. }
  945. /// <summary>
  946. /// 开启流程直接归档
  947. /// </summary>
  948. public async Task StartToEndAsync(StartWorkflowDto dto, string externalId, DateTime? expiredTime = null,
  949. CancellationToken cancellationToken = default)
  950. {
  951. var wfModule = await GetWorkflowModuleAsync(dto.DefinitionModuleCode, cancellationToken);
  952. var definition = wfModule.Definition;
  953. if (definition == null)
  954. throw new UserFriendlyException("无效模板编码");
  955. if (definition.Status is not EDefinitionStatus.Enable)
  956. throw new UserFriendlyException("该模板不可用");
  957. var endStepDefine = definition.FindEndStepDefine();
  958. dto.NextStepCode = endStepDefine.Code;
  959. dto.NextStepName = endStepDefine.Name;
  960. dto.FlowDirection = EFlowDirection.CenterToFile;
  961. await StartAsync(dto, externalId, expiredTime, cancellationToken: cancellationToken);
  962. }
  963. /// <summary>
  964. /// 查询派单池中流程节点id
  965. /// </summary>
  966. public async Task<IReadOnlyList<string>> GetUnhandleStepIdsFromSendPoolAsync(string sendPoolId,
  967. CancellationToken cancellationToken)
  968. {
  969. return await _workflowStepRepository.Queryable()
  970. .Where(d => SqlFunc.JsonListObjectAny(d.Handlers, "Key", sendPoolId))
  971. .Select(d => d.Id)
  972. .ToListAsync(cancellationToken);
  973. }
  974. /// <summary>
  975. /// 查询归属某用户的所有流程节点
  976. /// </summary>
  977. public async Task<List<WorkflowStep>> GetStepsBelongsToAsync(string userId, CancellationToken cancellationToken)
  978. {
  979. return await _workflowStepRepository.Queryable()
  980. .Includes(d => d.WorkflowTrace)
  981. .Where(d => d.HandlerId == userId)
  982. .OrderBy(d => d.CreationTime)
  983. .ToListAsync(cancellationToken);
  984. }
  985. /// <summary>
  986. /// 批量修改工单办理对象
  987. /// </summary>
  988. public async Task ChangeHandlerBatchAsync(
  989. IReadOnlyList<(string userId, string username, string orgId, string orgName, string? roleId, string? roleName, ICollection<WorkflowStep>
  990. steps)> handlers,
  991. CancellationToken cancellationToken)
  992. {
  993. foreach (var handler in handlers)
  994. {
  995. foreach (var step in handler.steps)
  996. {
  997. step.FlowAssignType = EFlowAssignType.User;
  998. step.Assign(handler.userId, handler.username,
  999. handler.orgId, handler.orgName, handler.roleId, handler.roleName);
  1000. if (step.WorkflowTrace is null)
  1001. throw new UserFriendlyException("未查询节点对应快照信息");
  1002. step.WorkflowTrace.FlowAssignType = EFlowAssignType.User;
  1003. step.WorkflowTrace.Assign(handler.userId, handler.username,
  1004. handler.orgId, handler.orgName, handler.roleId, handler.roleName);
  1005. //更新节点CreationTime 派单量统计 待派单数据使用
  1006. step.CreationTime = DateTime.Now;
  1007. }
  1008. }
  1009. var steps = handlers.SelectMany(d => d.steps).ToList();
  1010. //await _workflowStepRepository.UpdateRangeAsync(steps, cancellationToken);
  1011. await _workflowStepRepository.UpdateNav(steps)
  1012. .Include(d => d.WorkflowTrace)
  1013. .ExecuteCommandAsync();
  1014. }
  1015. /// <summary>
  1016. /// 查询工单办理中的一级部门
  1017. /// </summary>
  1018. public async Task<ICollection<Kv>> GetLevelOneOrgsAsync(string workflowId, CancellationToken cancellation)
  1019. {
  1020. var traces = await _workflowTraceRepository.Queryable()
  1021. .LeftJoin<SystemOrganize>((t, o) => t.HandlerOrgId == o.Id)
  1022. .Where((t, o) => t.WorkflowId == workflowId &&
  1023. !string.IsNullOrEmpty(t.HandlerOrgId) &&
  1024. o.Level == 1)
  1025. .ToListAsync(cancellation);
  1026. //var handlers = await _workflowStepHandlerRepository.Queryable()
  1027. // .InnerJoin<WorkflowTrace>((wsh, wt) => wsh.WorkflowStepId == wt.StepId)
  1028. // .LeftJoin<SystemOrganize>((wsh, wt, o) => wsh.OrgId == o.Id)
  1029. // .Where((wsh, wt, o) => wsh.WorkflowId == workflowId &&
  1030. // //wt.BusinessType == EBusinessType.Department &&
  1031. // //wt.HandlerType == EHandlerType.OrgLevel &&
  1032. // !string.IsNullOrEmpty(wsh.OrgId) &&
  1033. // o.Level == 1)
  1034. // .ToListAsync(cancellation);
  1035. //var orgs = handlers.Select(d => new Kv(d.OrgId, d.OrgName))
  1036. // .DistinctBy(d => d.Key)
  1037. // .ToList();
  1038. var orgs = traces
  1039. .DistinctBy(d => d.HandlerOrgId)
  1040. .Select(d => new Kv(d.HandlerOrgId, d.HandlerOrgName))
  1041. .ToList();
  1042. return orgs;
  1043. //var workflow = await GetWorkflowAsync(workflowId, withSteps: true, cancellationToken: cancellation);
  1044. //var list = workflow.Steps.Distinct().Where(d => d.BusinessType == EBusinessType.Department &&
  1045. // d.HandlerType == EHandlerType.OrgLevel &&
  1046. // d.StepHandlers.Any(d =>
  1047. // !string.IsNullOrEmpty(d.OrgId) && d.OrgId.CheckIfOrgLevelIs(1)))
  1048. // .Select(d => new Kv(d.StepHandlers.First().OrgId, d.StepHandlers.First().OrgName))
  1049. // .ToList();
  1050. //return list.Where((x, i) => list.FindIndex(z => z.Key == x.Key) == i).ToList();
  1051. }
  1052. /// <summary>
  1053. /// 更新未办理节点的期满时间
  1054. /// </summary>
  1055. public async Task UpdateUnhandleExpiredTimeAsync(string workflowId, DateTime expiredTime,
  1056. CancellationToken cancellation)
  1057. {
  1058. var steps = await _workflowStepRepository.Queryable()
  1059. .Includes(d => d.WorkflowTrace)
  1060. .Where(d => d.WorkflowId == workflowId &&
  1061. d.Status < EWorkflowStepStatus.Handled)
  1062. .ToListAsync(cancellation);
  1063. foreach (var step in steps)
  1064. {
  1065. step.StepExpiredTime = expiredTime;
  1066. step.WorkflowTrace.StepExpiredTime = expiredTime;
  1067. }
  1068. await _workflowStepRepository.UpdateNav(steps)
  1069. .Include(d => d.WorkflowTrace)
  1070. .ExecuteCommandAsync();
  1071. }
  1072. /// <summary>
  1073. /// 查询该部门最后办理节点
  1074. /// </summary>
  1075. /// <returns></returns>
  1076. public async Task<WorkflowStep> FindLastHandleStepAsync(string workflowId, string orgId, CancellationToken cancellation)
  1077. {
  1078. return await _workflowStepRepository.Queryable()
  1079. .Where(d => d.WorkflowId == workflowId
  1080. && d.HandlerOrgId == orgId
  1081. && d.StepType != EStepType.End
  1082. && d.StepType != EStepType.Summary
  1083. && !d.IsCountersignEndStep)
  1084. //.Where(d => d.StepHandlers.Any(sh => sh.OrgId == orgId) && d.WorkflowId == workflowId)
  1085. .OrderByDescending(d => d.HandleTime)
  1086. .FirstAsync(cancellation);
  1087. }
  1088. /// <summary>
  1089. /// 部门会签工单获取流程最顶级办理节点
  1090. /// </summary>
  1091. /// <returns></returns>
  1092. public async Task<WorkflowStep> FindTopHandleStepAsync(string workflowId, CancellationToken cancellation)
  1093. {
  1094. var workflow = await GetWorkflowAsync(workflowId, withSteps: true, cancellationToken: cancellation);
  1095. return workflow.Steps.FirstOrDefault(x => x.Id == workflow.TopCountersignStepId);
  1096. }
  1097. /// <summary>
  1098. /// 查询流转方向
  1099. /// </summary>
  1100. public EFlowDirection GetFlowDirection(EBusinessType sourceStepBusinessType,
  1101. EBusinessType directionStepBusinessType)
  1102. {
  1103. switch (sourceStepBusinessType)
  1104. {
  1105. case EBusinessType.Seat:
  1106. case EBusinessType.Send:
  1107. case EBusinessType.CenterMonitor:
  1108. case EBusinessType.CenterLeader:
  1109. return directionStepBusinessType switch
  1110. {
  1111. EBusinessType.Seat => EFlowDirection.CenterToCenter,
  1112. EBusinessType.Send => EFlowDirection.CenterToCenter,
  1113. EBusinessType.CenterMonitor => EFlowDirection.CenterToCenter,
  1114. EBusinessType.CenterLeader => EFlowDirection.CenterToCenter,
  1115. EBusinessType.Department => EFlowDirection.CenterToOrg,
  1116. EBusinessType.DepartmentLeader => EFlowDirection.CenterToOrg,
  1117. EBusinessType.File => EFlowDirection.CenterToFile,
  1118. _ => throw new ArgumentOutOfRangeException(nameof(directionStepBusinessType),
  1119. directionStepBusinessType, null)
  1120. };
  1121. case EBusinessType.Department:
  1122. case EBusinessType.DepartmentLeader:
  1123. return directionStepBusinessType switch
  1124. {
  1125. EBusinessType.Seat => EFlowDirection.OrgToCenter,
  1126. EBusinessType.Send => EFlowDirection.OrgToCenter,
  1127. EBusinessType.CenterMonitor => EFlowDirection.OrgToCenter,
  1128. EBusinessType.CenterLeader => EFlowDirection.OrgToCenter,
  1129. EBusinessType.Department => EFlowDirection.OrgToOrg,
  1130. EBusinessType.DepartmentLeader => EFlowDirection.OrgToOrg,
  1131. EBusinessType.File => EFlowDirection.OrgToFile,
  1132. _ => throw new ArgumentOutOfRangeException(nameof(directionStepBusinessType),
  1133. directionStepBusinessType, null)
  1134. };
  1135. case EBusinessType.File:
  1136. return directionStepBusinessType switch
  1137. {
  1138. EBusinessType.Seat => EFlowDirection.FiledToCenter,
  1139. EBusinessType.Send => EFlowDirection.FiledToCenter,
  1140. EBusinessType.Department => EFlowDirection.FiledToOrg,
  1141. EBusinessType.DepartmentLeader => EFlowDirection.FiledToOrg,
  1142. EBusinessType.CenterMonitor => EFlowDirection.FiledToCenter,
  1143. EBusinessType.CenterLeader => EFlowDirection.FiledToCenter,
  1144. _ => throw new ArgumentOutOfRangeException(nameof(directionStepBusinessType), directionStepBusinessType, null)
  1145. };
  1146. case EBusinessType.Unknown:
  1147. case EBusinessType.Publish:
  1148. case EBusinessType.Visit:
  1149. case EBusinessType.TrashEnd:
  1150. default:
  1151. throw new ArgumentOutOfRangeException(nameof(sourceStepBusinessType), sourceStepBusinessType, null);
  1152. }
  1153. }
  1154. /// <summary>
  1155. /// 流程被签收至某个用户(更新流转对象,办理对象,节点办理对象以及stepHandlers)
  1156. /// </summary>
  1157. public async Task<Workflow> SignToSomebodyAsync(string workflowId, string userId, string username, string orgId,
  1158. string orgName, CancellationToken cancellationToken)
  1159. {
  1160. var workflow = await GetWorkflowAsync(workflowId, withSteps: true, withTraces: true,
  1161. cancellationToken: cancellationToken);
  1162. var startStep = workflow.Steps.First(d => d.StepType == EStepType.Start && d.IsOrigin);
  1163. startStep.Handlers = new List<Kv> { new(userId, username) };
  1164. startStep.AcceptorId = userId;
  1165. startStep.AcceptorName = username;
  1166. startStep.AcceptTime = DateTime.Now;
  1167. startStep.AcceptorOrgId = orgId;
  1168. startStep.AcceptorOrgName = orgName;
  1169. startStep.FlowAssignType = EFlowAssignType.User;
  1170. startStep.Assign(userId, username, orgId, orgName);
  1171. startStep.WorkflowTrace = workflow.Traces.First(d => d.Id == startStep.Id);
  1172. _mapper.Map(startStep, startStep.WorkflowTrace);
  1173. await _workflowStepRepository.UpdateNav(startStep)
  1174. .Include(d => d.WorkflowTrace)
  1175. .ExecuteCommandAsync();
  1176. await _workflowRepository.UpdateAsync(workflow, cancellationToken);
  1177. return workflow;
  1178. }
  1179. /// <summary>
  1180. /// 非节点办理人员查询待办节点
  1181. /// </summary>
  1182. /// <returns></returns>
  1183. public async Task<ICollection<WorkflowStep>> GetUnhandleStepsByOthersAsync(string workflowId, CancellationToken cancellationToken)
  1184. {
  1185. return await _workflowStepRepository.Queryable()
  1186. .Where(d => d.WorkflowId == workflowId && d.Status != EWorkflowStepStatus.Handled)
  1187. .ToListAsync(cancellationToken);
  1188. }
  1189. /// <summary>
  1190. /// 根据汇总对象id找到被汇总节点,生成指派到用户的办理对象
  1191. /// </summary>
  1192. public StepAssignInfo GetSummaryTargetFlowStepHandler(Workflow workflow, string summaryTargetStepCode)
  1193. {
  1194. //根据汇总对象id找到被汇总节点
  1195. var summaryTargetStep = workflow.Steps.FirstOrDefault(d =>
  1196. d.StepType == EStepType.Normal &&
  1197. d.Code == summaryTargetStepCode &&
  1198. d.Status == EWorkflowStepStatus.Handled &&
  1199. d.IsOrigin);
  1200. if (summaryTargetStep is null)
  1201. throw UserFriendlyException.SameMessage("未查询到汇总对象节点");
  1202. var handler = new StepAssignInfo
  1203. {
  1204. Key = summaryTargetStep.HandlerId,
  1205. Value = summaryTargetStep.HandlerName,
  1206. UserId = summaryTargetStep.HandlerId,
  1207. Username = summaryTargetStep.HandlerName,
  1208. OrgId = summaryTargetStep.HandlerOrgId,
  1209. OrgName = summaryTargetStep.HandlerOrgName,
  1210. RoleId = summaryTargetStep.RoleId,
  1211. RoleName = summaryTargetStep.RoleName,
  1212. FlowAssignType = EFlowAssignType.User
  1213. };
  1214. return handler;
  1215. }
  1216. /// <summary>
  1217. /// 追加归档信息(接收ds推送12315归档信息)
  1218. /// </summary>
  1219. public async Task AppendFileOpinionAsync(string workflowId, string opinion, List<FileDto> files, CancellationToken cancellationToken)
  1220. {
  1221. //归档意见,附件追加在归档节点前一节点上
  1222. var workflow = await GetWorkflowAsync(workflowId, withSteps: true, withTraces: true, cancellationToken: cancellationToken);
  1223. var endStep = workflow.Steps.FirstOrDefault(d => d.StepType == EStepType.End);
  1224. if (endStep is null)
  1225. throw new UserFriendlyException($"该流程还未归档, workflowId: {workflowId}");
  1226. var prevStep = workflow.Steps.FirstOrDefault(d => d.Id == endStep.PrevStepId);
  1227. if (prevStep is null)
  1228. throw new UserFriendlyException($"未找到归档节点的前一节点, workflowId: {workflowId}, endStepId: {endStep.Id}");
  1229. prevStep.Opinion = opinion;
  1230. if (files != null && files.Any())
  1231. {
  1232. var filejsons = await _fileRepository.AddFileAsync(files, workflow.ExternalId, prevStep.Id, cancellationToken);
  1233. prevStep.FileJson.AddRange(filejsons);
  1234. }
  1235. await _workflowStepRepository.UpdateAsync(prevStep, cancellationToken);
  1236. var endTrace = workflow.Traces.FirstOrDefault(p => p.Id == prevStep.Id);
  1237. if (endTrace != null)
  1238. {
  1239. endTrace.Opinion = prevStep.Opinion;
  1240. endTrace.FileJson = prevStep.FileJson;
  1241. await _workflowTraceRepository.UpdateAsync(endTrace, cancellationToken);
  1242. }
  1243. }
  1244. /// <summary>
  1245. /// 跳转至结束节点(无视流程模板配置以及当前办理对象,直接跳至结束节点)
  1246. /// </summary>
  1247. public async Task JumpToEndAsync(ISessionContext current, string workflowId, string opinion, List<FileDto> files, DateTime? expiredTime,
  1248. EReviewResult reviewResult = EReviewResult.Unknown, CancellationToken cancellationToken = default)
  1249. {
  1250. var workflow = await GetWorkflowAsync(workflowId, withDefine: true, withSteps: true, withTraces: true,
  1251. withCountersigns: true, cancellationToken: cancellationToken);
  1252. await JumpToEndAsync(current, workflow, opinion, files, expiredTime, reviewResult, cancellationToken);
  1253. }
  1254. /// <summary>
  1255. /// 跳转至结束节点(无视流程模板配置以及当前办理对象,直接跳至结束节点)
  1256. /// </summary>
  1257. public async Task JumpToEndAsync(ISessionContext current, Workflow workflow, string opinion, List<FileDto> files, DateTime? expiredTime,
  1258. EReviewResult reviewResult = EReviewResult.Unknown, CancellationToken cancellationToken = default)
  1259. {
  1260. var endStepDefine = workflow.WorkflowDefinition.FindEndStepDefine();
  1261. if (endStepDefine is null)
  1262. throw new UserFriendlyException("未正确配置结束节点");
  1263. var dto = new BasicWorkflowDto
  1264. {
  1265. NextStepCode = endStepDefine.Code,
  1266. NextStepName = endStepDefine.Name,
  1267. FlowDirection = EFlowDirection.OrgToFile,
  1268. BusinessType = endStepDefine.BusinessType,
  1269. ReviewResult = reviewResult,
  1270. Opinion = opinion,
  1271. Files = files
  1272. };
  1273. var unhandleSteps = workflow.Steps
  1274. .Where(d => d.Status != EWorkflowStepStatus.Handled).ToList();
  1275. var unhandleTraces = workflow.Traces
  1276. .Where(d => d.Status != EWorkflowStepStatus.Handled).ToList();
  1277. //get currentStep
  1278. var currentStep = unhandleSteps.MaxBy(d => d.CreationTime)
  1279. ?? workflow.Steps.MaxBy(d => d.CreationTime);
  1280. foreach (var step in unhandleSteps)
  1281. {
  1282. await HandleStepAsync(step, workflow, dto, null, null, EHandleMode.Normal, cancellationToken);
  1283. if (step.IsStartCountersign)
  1284. step.CountersignEnd();
  1285. var trace = unhandleTraces.First(d => d.StepId == step.Id);
  1286. _mapper.Map(dto, trace);
  1287. _mapper.Map(step, trace);
  1288. }
  1289. await _workflowStepRepository.UpdateRangeAsync(unhandleSteps, cancellationToken);
  1290. await _workflowTraceRepository.UpdateRangeAsync(unhandleTraces, cancellationToken);
  1291. //结束会签
  1292. var counstersigns = workflow.Countersigns
  1293. .Where(d => !d.EndTime.HasValue)
  1294. .ToList();
  1295. foreach (var counstersign in counstersigns)
  1296. {
  1297. //结束会签
  1298. counstersign.End(currentStep.Id, currentStep.Code, currentStep.BusinessType,
  1299. current.UserId, current.UserName,
  1300. current.OrgId, current.OrgName,
  1301. current.OrgAreaCode, current.OrgAreaName);
  1302. }
  1303. await _workflowCountersignRepository.UpdateRangeAsync(counstersigns, cancellationToken);
  1304. //更新实际办理节点信息
  1305. if (currentStep.StepType != EStepType.Summary && currentStep.StepType != EStepType.End)
  1306. {
  1307. workflow.UpdateActualStepWhenHandle(currentStep, current.OrgAreaCode, current.OrgAreaName, current.OrgLevel);
  1308. workflow.ActualHandleStepAcceptTime = currentStep.AcceptTime;
  1309. }
  1310. //
  1311. // workflow.UpdateCurrentStepWhenHandle(currentStep, current.OrgAreaCode, current.OrgAreaName, current.OrgLevel);
  1312. if (workflow.Steps.All(d => d.StepType != EStepType.End))
  1313. {
  1314. await EndAsync(current, workflow, dto, endStepDefine, currentStep, expiredTime, cancellationToken);
  1315. }
  1316. }
  1317. /// <summary>
  1318. /// 查找当前会签内所有节点(含start,end)
  1319. /// </summary>
  1320. private void SearchCountersignSteps(WorkflowStep startStep, List<WorkflowStep> steps,
  1321. ref List<WorkflowStep> csSteps)
  1322. {
  1323. if (startStep.IsStartCountersign)
  1324. {
  1325. var countersignSteps = steps.Where(d => d.CountersignId == startStep.StartCountersignId).ToList();
  1326. if (countersignSteps.Any())
  1327. {
  1328. foreach (var countersignStep in countersignSteps)
  1329. {
  1330. SearchCountersignSteps(countersignStep, steps, ref csSteps);
  1331. }
  1332. }
  1333. }
  1334. csSteps.Add(startStep);
  1335. }
  1336. /// <summary>
  1337. /// 撤回(返回到之前任意节点)
  1338. /// </summary>
  1339. public async Task<(Workflow, StepDefine, WorkflowStep, WorkflowStep, WorkflowStep, bool)> RecallAsync(
  1340. RecallDto dto, StepAssignInfo stepAssignInfo,
  1341. EWorkflowTraceType traceType, DateTime? expiredTime, bool isOrderFiled, EHandleMode handleMode,
  1342. Action<Workflow, WorkflowStep, StepDefine, WorkflowStep, WorkflowStep>? newStepConfig = null,
  1343. CancellationToken cancellationToken = default)
  1344. {
  1345. var workflow = await GetWorkflowAsync(dto.WorkflowId, withDefine: true, withSteps: true,
  1346. withTraces: true, withCountersigns: true, cancellationToken: cancellationToken);
  1347. var targetStepDefine = GetStepDefine(workflow.WorkflowDefinition, dto.NextStepCode);
  1348. if (targetStepDefine.StepType is EStepType.End)
  1349. throw UserFriendlyException.SameMessage("结束节点不支持撤回");
  1350. var targetStep = workflow.Steps.FirstOrDefault(d => d.Code == dto.NextStepCode && d.IsOrigin);
  1351. if (targetStep is null)
  1352. throw UserFriendlyException.SameMessage("该流程尚未流转至该节点");
  1353. return await RecallAsync(workflow, dto, stepAssignInfo, targetStepDefine, targetStep, traceType,
  1354. expiredTime, isOrderFiled, handleMode, newStepConfig, cancellationToken);
  1355. }
  1356. /// <summary>
  1357. /// 撤回(返回到之前任意节点)
  1358. /// </summary>
  1359. public Task<(Workflow, StepDefine, WorkflowStep, WorkflowStep, WorkflowStep, bool)> RecallAsync(
  1360. Workflow workflow, BasicWorkflowDto dto, StepAssignInfo stepAssignInfo,
  1361. StepDefine targetStepDefine, EWorkflowTraceType traceType, DateTime? expiredTime, bool isOrderFiled, EHandleMode handleMode,
  1362. Action<Workflow, WorkflowStep, StepDefine, WorkflowStep, WorkflowStep>? newStepConfig = null,
  1363. CancellationToken cancellationToken = default)
  1364. {
  1365. var targetStep = workflow.Steps.FirstOrDefault(d => d.Code == dto.NextStepCode && d.IsOrigin);
  1366. if (targetStep is null)
  1367. throw UserFriendlyException.SameMessage("该流程尚未流转至该节点");
  1368. return RecallAsync(workflow, dto, stepAssignInfo, targetStepDefine, targetStep,
  1369. traceType, expiredTime, isOrderFiled, handleMode, newStepConfig, cancellationToken);
  1370. }
  1371. public Task<(Workflow, StepDefine, WorkflowStep, WorkflowStep, WorkflowStep, bool)> RecallAsync(
  1372. Workflow workflow, BasicWorkflowDto dto, StepAssignInfo stepAssignInfo,
  1373. WorkflowStep targetStep, EWorkflowTraceType traceType, DateTime? expiredTime, bool isOrderFiled, EHandleMode handleMode,
  1374. Action<Workflow, WorkflowStep, StepDefine, WorkflowStep, WorkflowStep>? newStepConfig = null,
  1375. CancellationToken cancellationToken = default)
  1376. {
  1377. var targetStepDefine = GetStepDefine(workflow.WorkflowDefinition, dto.NextStepCode);
  1378. if (targetStepDefine.StepType is EStepType.End)
  1379. throw UserFriendlyException.SameMessage("结束节点不支持撤回");
  1380. return RecallAsync(workflow, dto, stepAssignInfo,
  1381. targetStepDefine, targetStep, traceType,
  1382. expiredTime, isOrderFiled, handleMode, newStepConfig, cancellationToken);
  1383. }
  1384. /// <summary>
  1385. /// 特提
  1386. /// </summary>
  1387. /// <returns>workflow, targetStepDefine, currentStep, targetStep, newStep, isOrgToCenter</returns>
  1388. /// <exception cref="UserFriendlyException"></exception>
  1389. private async Task<(Workflow workflow, StepDefine targetStepDefine,
  1390. WorkflowStep currentStep, WorkflowStep targetStep, WorkflowStep newStep, bool isOrgToCenter)>
  1391. RecallAsync(
  1392. Workflow workflow, BasicWorkflowDto dto, StepAssignInfo stepAssignInfo,
  1393. StepDefine targetStepDefine, WorkflowStep targetStep, EWorkflowTraceType traceType,
  1394. DateTime? expiredTime, bool isOrderFiled, EHandleMode handleMode,
  1395. Action<Workflow, WorkflowStep, StepDefine, WorkflowStep, WorkflowStep>? newStepConfig = null,
  1396. CancellationToken cancellationToken = default)
  1397. {
  1398. var targetIsStartStep = targetStepDefine.StepType is EStepType.Start;
  1399. var updateTraces = new List<WorkflowTrace>();
  1400. var currentStep = workflow.Steps.Where(x => x.Status != EWorkflowStepStatus.Handled)
  1401. .MaxBy(d => d.CreationTime) ??
  1402. workflow.Steps.Where(x => x.StepType == EStepType.End)
  1403. .MaxBy(d => d.CreationTime) ??
  1404. workflow.Steps.MaxBy(d => d.CreationTime);
  1405. //update uncomplete traces
  1406. var uncompleteTraces = workflow.Traces.Where(d => d.Status != EWorkflowStepStatus.Handled && d.TraceStyle == ETraceStyle.Flow).ToList();
  1407. if (uncompleteTraces.Any())
  1408. {
  1409. foreach (var trace in uncompleteTraces)
  1410. {
  1411. trace.Handle(
  1412. _sessionContext.RequiredUserId, _sessionContext.UserName,
  1413. _sessionContext.RequiredOrgId, _sessionContext.OrgName,
  1414. _sessionContext.OrgAreaCode, _sessionContext.OrgAreaName,
  1415. _sessionContext.OrgIsCenter, handleMode, dto.Opinion);
  1416. }
  1417. //await _workflowTraceRepository.UpdateRangeAsync(uncompleteTraces, cancellationToken);
  1418. updateTraces.AddRange(uncompleteTraces);
  1419. }
  1420. else
  1421. {
  1422. var endTrace = workflow.Traces.Where(d => d.StepType == EStepType.End && d.TraceStyle == ETraceStyle.Flow).MaxBy(d => d.CreationTime);
  1423. if (endTrace is not null)
  1424. {
  1425. endTrace.Opinion += ("\r\n" + dto.Opinion);
  1426. updateTraces.Add(endTrace);
  1427. }
  1428. }
  1429. //get targetStep's previous
  1430. WorkflowStep? targetPrevStep = null;
  1431. if (!targetIsStartStep)
  1432. {
  1433. targetPrevStep = workflow.Steps.FirstOrDefault(d => d.Id == targetStep.PrevStepId);
  1434. if (targetPrevStep == null)
  1435. throw new UserFriendlyException($"{nameof(RecallAsync)}, 未找到目标节点的前一节点, flowId: {workflow.Id}");
  1436. }
  1437. //查询所有目标节点之后的节点,然后删掉(包括目标节点)
  1438. var removeSteps = GetStepsBehindTargetStep(workflow.Steps, targetStep);
  1439. if (removeSteps.Any())
  1440. {
  1441. await _workflowStepRepository.RemoveRangeAsync(removeSteps, cancellationToken);
  1442. workflow.Steps.RemoveAll(d => removeSteps.Contains(d));
  1443. //更新快照对应节点状态
  1444. var stepIds = removeSteps.Select(d => d.Id).ToList();
  1445. var traces = workflow.Traces.Where(d => stepIds.Contains(d.StepId)).ToList();
  1446. //await UpdateTracesStateAsync(updateTraces, EWorkflowTraceState.StepRemoveByRecall, cancellationToken);
  1447. foreach (var trace in traces)
  1448. {
  1449. trace.TraceState = isOrderFiled
  1450. ? EWorkflowTraceState.StepRemoveByRecallWhenFiled
  1451. : EWorkflowTraceState.StepRemoveByRecall;
  1452. }
  1453. updateTraces.AddRange(traces);
  1454. }
  1455. await _workflowTraceRepository.UpdateRangeAsync(updateTraces, cancellationToken);
  1456. //结束会签
  1457. var unCompleteCountersigns = workflow.Countersigns.Where(d => !d.IsCompleted()).ToList();
  1458. if (unCompleteCountersigns.Any())
  1459. {
  1460. foreach (var unCompleteCountersign in unCompleteCountersigns)
  1461. {
  1462. unCompleteCountersign.End(null, null, EBusinessType.File,
  1463. _sessionContext.RequiredUserId, _sessionContext.UserName,
  1464. _sessionContext.RequiredOrgId, _sessionContext.OrgName,
  1465. _sessionContext.OrgAreaCode, _sessionContext.OrgAreaName);
  1466. }
  1467. await _workflowCountersignRepository.UpdateRangeAsync(unCompleteCountersigns, cancellationToken);
  1468. }
  1469. workflow.EndCountersign();
  1470. workflow.ResetOption();
  1471. if (workflow.Status is EWorkflowStatus.Completed)
  1472. workflow.SetStatusRunnable();
  1473. var assigner = new UserInfo(
  1474. _sessionContext.UserId,
  1475. _sessionContext.UserName,
  1476. _sessionContext.OrgId,
  1477. _sessionContext.OrgName,
  1478. _sessionContext.OrgIsCenter
  1479. );
  1480. //var targetStepNew = targetIsStartStep
  1481. // ? await CreateStartStepAsync(workflow, targetStepDefine, dto, assigner,
  1482. // dto.NextHandlers.First(), traceType, expiredTime, flowAssignInfo.FlowAssignType, cancellationToken)
  1483. // : (await CreateStepsAsync(workflow, targetStepDefine, targetPrevStep, dto, assigner,
  1484. // flowAssignInfo.FlowAssignType, dto.NextHandlers,
  1485. // null, EWorkflowStepStatus.WaitForAccept, ECountersignPosition.None, true, traceType,
  1486. // null, expiredTime, cancellationToken: cancellationToken)).First();
  1487. //var stepAssignInfo = GetStepAssignInfo(reverseFlowStepAssignInfo, targetStep, targetStepDefine);
  1488. stepAssignInfo ??= targetStep.GetWorkflowStepHandler();
  1489. WorkflowStep targetStepNew;
  1490. if (targetIsStartStep)
  1491. {
  1492. // targetStepNew = await CreateStartStepAsync(workflow, targetStepDefine, dto, assigner,
  1493. // stepAssignInfo, traceType, expiredTime, stepConfig, cancellationToken);
  1494. targetStepNew = CreateStartStep(workflow, targetStepDefine, dto, assigner, stepAssignInfo, expiredTime);
  1495. //await Task.Run(() => newStepConfig?.Invoke(workflow, currentStep, targetStepDefine, targetStep, targetStepNew), cancellationToken);
  1496. newStepConfig?.Invoke(workflow, currentStep, targetStepDefine, targetStep, targetStepNew);
  1497. await _workflowStepRepository.AddAsync(targetStepNew, cancellationToken);
  1498. workflow.Steps.Add(targetStepNew);
  1499. await CreateTraceAsync(workflow, targetStepNew, traceType, cancellationToken);
  1500. }
  1501. else
  1502. {
  1503. // targetStepNew = (await CreateStepsAsync(workflow, targetStepDefine, targetPrevStep, dto, assigner, [stepAssignInfo],
  1504. // null, EWorkflowStepStatus.WaitForAccept, ECountersignPosition.None, true, traceType,
  1505. // null, expiredTime, stepConfig: stepConfig, cancellationToken: cancellationToken)).First();
  1506. targetStepNew = CreateStep(workflow, targetStepDefine, targetPrevStep, stepAssignInfo, assigner,
  1507. dto.NextStepCode, null, EWorkflowStepStatus.WaitForAccept, ECountersignPosition.None, expiredTime,
  1508. dto.NextStepName, true, true, null, dto.BusinessType, dto.FlowDirection);
  1509. //await Task.Run(() => newStepConfig?.Invoke(workflow, currentStep, targetStepDefine, targetStep, targetStepNew), cancellationToken);
  1510. newStepConfig?.Invoke(workflow, currentStep, targetStepDefine, targetStep, targetStepNew);
  1511. await _workflowStepRepository.AddAsync(targetStepNew, cancellationToken);
  1512. workflow.Steps.Add(targetStepNew);
  1513. await CreateTraceAsync(workflow, targetStepNew, traceType, cancellationToken);
  1514. }
  1515. //更新实际办理节点信息
  1516. workflow.UpdateActualStepWhenAssign(targetStepNew, new FlowStepHandler
  1517. {
  1518. UserId = targetStep.HandlerId,
  1519. Username = targetStep.HandlerName,
  1520. OrgId = targetStep.HandlerOrgId,
  1521. OrgName = targetStep.HandlerOrgName
  1522. });
  1523. workflow.UpdateCurrentStepWhenAssign(targetStepNew, new FlowStepHandler
  1524. {
  1525. UserId = targetStep.HandlerId,
  1526. Username = targetStep.HandlerName,
  1527. OrgId = targetStep.HandlerOrgId,
  1528. OrgName = targetStep.HandlerOrgName
  1529. });
  1530. //取消维护workflow得冗余字段(FlowedOrgIds, HandlerOrgs)
  1531. //workflow.Assign(flowAssignInfo.FlowAssignType, flowAssignInfo.GetHandlerIds());
  1532. //workflow.ResetHandlers(flowAssignInfo.FlowAssignType, flowAssignInfo.HandlerObjects);
  1533. await _workflowRepository.UpdateAsync(workflow, cancellationToken);
  1534. //calc workflow expired time
  1535. var isOrgToCenter = CheckIfFlowFromOrgToCenter(workflow, targetStep);
  1536. await _publisher.PublishAsync(new RecallNotify(workflow, targetStep, dto, isOrgToCenter),
  1537. PublishStrategy.ParallelWhenAll, cancellationToken);
  1538. return (workflow, targetStepDefine, currentStep, targetStep, targetStepNew, isOrgToCenter);
  1539. }
  1540. /// <summary>
  1541. /// 撤回至开始节点
  1542. /// </summary>
  1543. public async Task RecallToStartStepAsync(Workflow workflow, string opinion,
  1544. DateTime? expiredTime, bool isOrderFiled, EHandleMode handleMode, EFlowAssignType? flowAssignType,
  1545. CancellationToken cancellationToken = default)
  1546. {
  1547. //todo 1.当前待办节点删掉 2.当前待办trace更新(status, opinion) 3.复制startStep为待办 4.更新workflow(status, csStatus, handlers) 5.publish event
  1548. var startStep = workflow.Steps.Where(d => d.StepType == EStepType.Start && d.IsOrigin)
  1549. .MaxBy(d => d.CreationTime);
  1550. if (startStep is null)
  1551. throw new UserFriendlyException($"数据异常, workflowId: {workflow.Id}", "该流程无开始节点");
  1552. //await RecallToTargetStepAsync(workflow, startStep, opinion, current, cancellationToken);
  1553. var startStepDefine = workflow.WorkflowDefinition.Steps.FirstOrDefault(d => d.StepType == EStepType.Start);
  1554. var dto = new RecallDto
  1555. {
  1556. Opinion = opinion,
  1557. NextStepCode = startStep.Code,
  1558. NextStepName = startStep.Name,
  1559. BusinessType = startStep.BusinessType,
  1560. StepType = startStep.StepType,
  1561. HandlerType = startStepDefine.HandlerType,
  1562. NextHandlers = new List<StepAssignInfo>
  1563. {
  1564. flowAssignType == EFlowAssignType.User
  1565. ? new()
  1566. {
  1567. Key = startStep.HandlerId,
  1568. Value = startStep.HandlerName,
  1569. RoleId = startStep.RoleId,
  1570. RoleName = startStep.RoleName,
  1571. UserId = startStep.HandlerId,
  1572. Username = startStep.HandlerName,
  1573. OrgId = startStep.HandlerOrgId,
  1574. OrgName = startStep.HandlerOrgName,
  1575. FlowAssignType = EFlowAssignType.User
  1576. }
  1577. : new()
  1578. {
  1579. Key = startStep.RoleId,
  1580. Value = startStep.RoleName,
  1581. RoleId = startStep.RoleId,
  1582. RoleName = startStep.RoleName,
  1583. UserId = startStep.HandlerId,
  1584. Username = startStep.HandlerName,
  1585. OrgId = startStep.HandlerOrgId,
  1586. OrgName = startStep.HandlerOrgName,
  1587. FlowAssignType = EFlowAssignType.Role
  1588. }
  1589. }
  1590. };
  1591. // var flowAssignInfo = await GetNextStepFlowAssignInfoByDefineAsync(targetStepDefine, dto.HandlerType, dto.IsStartCountersign,
  1592. // dto.NextHandlers.Select(d => new Kv(d.Key, d.Value)).ToList(), cancellationToken);
  1593. // //flowAssignInfo.FlowAssignType = EFlowAssignType.Role;
  1594. // await RecallAsync(workflow, dto, targetStepDefine, flowAssignInfo, EWorkflowTraceType.Recall, expiredTime, isOrderFiled,
  1595. // handleMode, cancellationToken);
  1596. await RecallAsync(workflow, dto, dto.NextHandlers.First(), startStepDefine, startStep,
  1597. EWorkflowTraceType.Recall, expiredTime, isOrderFiled, handleMode, cancellationToken: cancellationToken);
  1598. }
  1599. /// <summary>
  1600. /// 撤回至派单节点
  1601. /// </summary>
  1602. public async Task RecallToSendStepAsync(Workflow workflow, string opinion,
  1603. DateTime? expiredTime, bool isOrderFiled, EHandleMode handleMode, EFlowAssignType? flowAssignType,
  1604. CancellationToken cancellationToken = default)
  1605. {
  1606. var sendStep = workflow.Steps.Where(d => d.BusinessType == EBusinessType.Send && d.IsOrigin)
  1607. .MaxBy(d => d.CreationTime);
  1608. if (sendStep is null)
  1609. throw new UserFriendlyException($"未找到派单节点, workflowId: {workflow.Id}", "该流程无派单节点");
  1610. //await RecallToTargetStepAsync(workflow, sendStep, opinion, current, cancellationToken);
  1611. var sendStepDefine = workflow.WorkflowDefinition.Steps.FirstOrDefault(d => d.BusinessType == EBusinessType.Send);
  1612. var dto = new RecallDto
  1613. {
  1614. Opinion = opinion,
  1615. NextStepCode = sendStep.Code,
  1616. NextStepName = sendStep.Name,
  1617. BusinessType = sendStep.BusinessType,
  1618. StepType = sendStep.StepType,
  1619. HandlerType = sendStepDefine.HandlerType,
  1620. NextHandlers = new List<StepAssignInfo>
  1621. {
  1622. flowAssignType == EFlowAssignType.User
  1623. ? new()
  1624. {
  1625. Key = sendStep.HandlerId,
  1626. Value = sendStep.HandlerName,
  1627. RoleId = sendStep.RoleId,
  1628. RoleName = sendStep.RoleName,
  1629. UserId = sendStep.HandlerId,
  1630. Username = sendStep.HandlerName,
  1631. OrgId = sendStep.HandlerOrgId,
  1632. OrgName = sendStep.HandlerOrgName,
  1633. FlowAssignType = EFlowAssignType.User
  1634. }
  1635. : new()
  1636. {
  1637. Key = sendStep.RoleId,
  1638. Value = sendStep.RoleName,
  1639. RoleId = sendStep.RoleId,
  1640. RoleName = sendStep.RoleName,
  1641. UserId = sendStep.HandlerId,
  1642. Username = sendStep.HandlerName,
  1643. OrgId = sendStep.HandlerOrgId,
  1644. OrgName = sendStep.HandlerOrgName,
  1645. FlowAssignType = EFlowAssignType.Role
  1646. }
  1647. }
  1648. };
  1649. //var flowAssignInfo = await GetNextStepFlowAssignInfoByDefineAsync(targetStepDefine, dto.HandlerType, dto.IsStartCountersign,
  1650. // dto.NextHandlers.Select(d => new Kv(d.Key, d.Value)).ToList(), cancellationToken);
  1651. ////flowAssignInfo.FlowAssignType = EFlowAssignType.Role;
  1652. //await RecallAsync(workflow, dto, targetStepDefine, flowAssignInfo, EWorkflowTraceType.Recall, expiredTime, isOrderFiled,
  1653. // handleMode, cancellationToken);
  1654. await RecallAsync(workflow, dto, dto.NextHandlers.First(), sendStepDefine, sendStep,
  1655. EWorkflowTraceType.Recall, expiredTime, isOrderFiled, handleMode, cancellationToken: cancellationToken);
  1656. }
  1657. /// <summary>
  1658. /// 特提至中心(优先派单组其次坐席)
  1659. /// </summary>
  1660. /// <returns>true 派单组 false 话务部</returns>
  1661. public async Task<(bool, Workflow workflow)> RecallToCenterFirstToSendAsync(string workflowId, string opinion,
  1662. bool isOrderFiled, DateTime? expiredTime, EHandleMode handleMode, EFlowAssignType? flowAssignType = EFlowAssignType.User,
  1663. CancellationToken cancellationToken = default)
  1664. {
  1665. var isPaiDan = false;
  1666. var workflow = await GetWorkflowAsync(workflowId, withDefine: true, withSteps: true, withTraces: true, withCountersigns: true,
  1667. cancellationToken: cancellationToken);
  1668. var hasSendStep = workflow.Steps.Any(d => d.BusinessType == EBusinessType.Send);
  1669. if (hasSendStep)
  1670. {
  1671. await RecallToSendStepAsync(workflow, opinion, expiredTime, isOrderFiled, handleMode, flowAssignType, cancellationToken);
  1672. isPaiDan = true;
  1673. }
  1674. else
  1675. {
  1676. await RecallToStartStepAsync(workflow, opinion, expiredTime, isOrderFiled, handleMode, flowAssignType, cancellationToken);
  1677. //isPaiDan = false;
  1678. }
  1679. return (new(isPaiDan, workflow));
  1680. }
  1681. /// <summary>
  1682. /// 补充
  1683. /// </summary>
  1684. /// <returns></returns>
  1685. public async Task SupplementAsync(Workflow workflow, EndWorkflowDto dto, CancellationToken cancellationToken)
  1686. {
  1687. CheckWhetherRunnable(workflow.Status);
  1688. //todo 检查当前办理人是否为该流程中的办理人
  1689. var supplement = _mapper.Map<WorkflowSupplement>(dto);
  1690. //await _workflowSupplementRepository.AddAsync(supplement, cancellationToken);
  1691. }
  1692. /// <summary>
  1693. /// 终止流程
  1694. /// </summary>
  1695. public async Task TerminateAsync(TerminateDto dto, CancellationToken cancellationToken)
  1696. {
  1697. var workflow = await _workflowRepository.GetAsync(dto.WorkflowId, cancellationToken);
  1698. if (workflow == null)
  1699. throw UserFriendlyException.SameMessage("无效的流程编号");
  1700. workflow.Terminate(dto.Opinion);
  1701. await _workflowRepository.UpdateAsync(workflow, cancellationToken);
  1702. await _publisher.PublishAsync(new TerminalWorkflowNotify(workflow), PublishStrategy.ParallelWhenAll,
  1703. cancellationToken);
  1704. }
  1705. /// <summary>
  1706. /// 根据stepCode查询流程配置中对应的节点
  1707. /// </summary>
  1708. public StepDefine GetStepDefine(WorkflowDefinition workflowDefinition, string stepCode)
  1709. {
  1710. if (workflowDefinition == null) throw new ArgumentNullException(nameof(workflowDefinition));
  1711. if (string.IsNullOrEmpty(stepCode)) throw new ArgumentNullException(nameof(stepCode));
  1712. var stepDefine = workflowDefinition.FindStepDefine(stepCode);
  1713. if (stepDefine == null)
  1714. throw new UserFriendlyException(
  1715. $"未找到流程中对应的节点,DefineCode: {workflowDefinition.Code}, stepCode: {stepCode}",
  1716. "未查询到对应节点");
  1717. return stepDefine;
  1718. }
  1719. /// <summary>
  1720. /// 查询当前待办理节点
  1721. /// </summary>
  1722. public WorkflowStep FindCurrentStepWaitForHandle(Workflow workflow, string userId, string orgId,
  1723. string[] roleIds) =>
  1724. GetUnHandleStep(workflow.Steps, orgId, userId, roleIds);
  1725. /// <summary>
  1726. /// 查询当前节点中最后一个节点
  1727. /// </summary>
  1728. public async Task<WorkflowStep?> FindLastStepAsync(string workflowId, CancellationToken cancellationToken)
  1729. {
  1730. var workflow = await GetWorkflowAsync(workflowId, withSteps: true, cancellationToken: cancellationToken);
  1731. return workflow.Steps.MaxBy(d => d.CreationTime);
  1732. }
  1733. /// <summary>
  1734. /// 查询所有办理部门及实际办理部门
  1735. /// </summary>
  1736. /// <returns></returns>
  1737. public async Task<(Kv, IReadOnlyList<Kv>)> GetHandleOrgsAsync(string workflowId,
  1738. CancellationToken cancellationToken)
  1739. {
  1740. var workflow = await GetWorkflowAsync(workflowId, withTraces: true, cancellationToken: cancellationToken);
  1741. var steps = workflow.Traces
  1742. .Where(d => d.StepType is EStepType.Normal)
  1743. .ToList();
  1744. var items = steps.Where(d => d.TraceType == EWorkflowTraceType.Normal || d.TraceType == EWorkflowTraceType.Jump)
  1745. .Select(d => new Kv(d.HandlerOrgId, d.HandlerOrgName))
  1746. .DistinctBy(d => d.Key).ToList();
  1747. return (new Kv(workflow.ActualHandleOrgCode, workflow.ActualHandleOrgName), items);
  1748. }
  1749. /// <summary>
  1750. /// 新增流程流转记录
  1751. /// </summary>
  1752. public async Task AddStepsAsync(string workflowId, List<WorkflowStep> steps, CancellationToken cancellationToken)
  1753. {
  1754. var workflow = await GetWorkflowAsync(workflowId, cancellationToken: cancellationToken);
  1755. if (workflow is null)
  1756. throw new UserFriendlyException("找不到该流程");
  1757. await _workflowStepRepository.AddNav(steps)
  1758. .Include(d => d.WorkflowTrace)
  1759. .ExecuteCommandAsync();
  1760. }
  1761. /// <summary>
  1762. /// 创建开始节点
  1763. /// </summary>
  1764. public WorkflowStep CreateStartStep(
  1765. Workflow workflow, StepDefine startStepDefine,
  1766. BasicWorkflowDto dto, UserInfo assigner, StepAssignInfo handler, DateTime? expiredTime,
  1767. //EFlowAssignType? flowAssignType = EFlowAssignType.User,
  1768. Action<Workflow, WorkflowStep, StepDefine, WorkflowStep>? newStepConfig = null)
  1769. {
  1770. //startstep
  1771. var nextSteps = _mapper.Map<List<StepSimple>>(startStepDefine.NextSteps);
  1772. if (startStepDefine.InstanceMode is EInstanceMode.Config)
  1773. {
  1774. var selectedStep = nextSteps.FirstOrDefault(d => d.Code == dto.NextStepCode);
  1775. if (selectedStep is not null)
  1776. selectedStep.Selected = true;
  1777. }
  1778. var startStep = _mapper.Map<WorkflowStep>(startStepDefine);
  1779. _mapper.Map(workflow, startStep);
  1780. //startStep.FlowAssignType = flowAssignType;
  1781. startStep.Handlers = new List<Kv> { new(handler.Key, handler.Value) };
  1782. startStep.NextSteps = nextSteps;
  1783. startStep.IsMain = true;
  1784. startStep.IsOrigin = true;
  1785. startStep.Status = EWorkflowStepStatus.WaitForAccept;
  1786. startStep.FlowDirection = dto.FlowDirection;
  1787. startStep.PrevChosenStepCode = null;
  1788. if (expiredTime.HasValue)
  1789. startStep.StepExpiredTime = expiredTime;
  1790. startStep.Assign(handler);
  1791. startStep.AssignerId = assigner.UserId;
  1792. startStep.AssignerName = assigner.UserName;
  1793. startStep.AssignerOrgId = assigner.OrgId;
  1794. startStep.AssignerOrgName = assigner.OrgName;
  1795. startStep.AssignerOrgIsCenter = assigner.OrgIsCenter;
  1796. startStep.InitId();
  1797. newStepConfig?.Invoke(workflow, null, startStepDefine, startStep);
  1798. return startStep;
  1799. }
  1800. /// <summary>
  1801. /// 流程结束
  1802. /// </summary>
  1803. public async Task<(WorkflowStep, WorkflowTrace)> EndAsync(ISessionContext current, Workflow workflow, BasicWorkflowDto dto,
  1804. StepDefine endStepDefine, WorkflowStep currentStep,
  1805. DateTime? expiredTime, CancellationToken cancellationToken)
  1806. {
  1807. //create endStep
  1808. var endStep = await CreateEndStepAsync(workflow, endStepDefine, currentStep, dto, expiredTime,
  1809. cancellationToken);
  1810. //workflow.Steps.Add(endStep);
  1811. //update endTrace
  1812. var endTrace = await NextTraceAsync(workflow, dto, endStep, cancellationToken);
  1813. //create publish trace
  1814. if (workflow.FlowType is EFlowType.Handle)
  1815. await CreatePublishTraceAsync(endTrace, cancellationToken);
  1816. workflow.Complete(endStep, dto.ReviewResult);
  1817. //需求调整:归档时当前节点显示为归档节点
  1818. workflow.UpdateCurrentStepWhenHandle(endStep, current.OrgAreaCode, current.OrgAreaName, current.OrgLevel);
  1819. workflow.CurrentStepAcceptTime = endStep.AcceptTime;
  1820. // workflow.UpdateActualStepWhenHandle(endStep, current.OrgAreaCode, current.OrgAreaName, current.OrgLevel);
  1821. // workflow.ActualHandleStepAcceptTime = endStep.AcceptTime.Value;
  1822. if (string.IsNullOrEmpty(workflow.OrgLevelOneCode))
  1823. workflow.UpdateLevelOneOrg(workflow.ActualHandleOrgCode, workflow.ActualHandleOrgName);
  1824. await _workflowRepository.UpdateAsync(workflow, cancellationToken);
  1825. await _publisher.PublishAsync(new EndWorkflowNotify(workflow, endTrace, dto),
  1826. PublishStrategy.ParallelWhenAll, cancellationToken);
  1827. return (endStep, endTrace);
  1828. }
  1829. ///// <summary>
  1830. ///// 判断会签类型(中心会签或部门会签)
  1831. ///// </summary>
  1832. ///// <param name="businessType"></param>
  1833. ///// <returns></returns>
  1834. ///// <exception cref="ArgumentOutOfRangeException"></exception>
  1835. //public ECounterSignType? GetCounterSignType(EBusinessType businessType) =>
  1836. // businessType switch
  1837. // {
  1838. // EBusinessType.Seat => ECounterSignType.Seat,
  1839. // EBusinessType.Send => ECounterSignType.Seat,
  1840. // EBusinessType.Department => ECounterSignType.Department,
  1841. // EBusinessType.File => null,
  1842. // _ => throw new ArgumentOutOfRangeException(nameof(businessType), businessType, null)
  1843. // };
  1844. public ECounterSignType? GetCounterSignType(bool isStartCountersign, EBusinessType currentStepBusinessType)
  1845. {
  1846. if (!isStartCountersign) return null;
  1847. return currentStepBusinessType switch
  1848. {
  1849. EBusinessType.Seat => ECounterSignType.Center,
  1850. EBusinessType.Send => ECounterSignType.Center,
  1851. EBusinessType.CenterMonitor => ECounterSignType.Center,
  1852. EBusinessType.CenterLeader => ECounterSignType.Center,
  1853. EBusinessType.Department => ECounterSignType.Department,
  1854. EBusinessType.DepartmentLeader => ECounterSignType.Department,
  1855. EBusinessType.File => null,
  1856. _ => throw new ArgumentOutOfRangeException(nameof(currentStepBusinessType), currentStepBusinessType, null)
  1857. };
  1858. // return _sessionContext.OrgIsCenter ? ECounterSignType.Center : ECounterSignType.Department;
  1859. }
  1860. /// <summary>
  1861. /// 办理节点
  1862. /// </summary>
  1863. public async Task HandleStepAsync(WorkflowStep step, Workflow workflow,
  1864. BasicWorkflowDto dto, ECounterSignType? counterSignType,
  1865. DateTime? expiredTime, EHandleMode handleMode, CancellationToken cancellationToken)
  1866. {
  1867. if (step.Status is EWorkflowStepStatus.Handled)
  1868. throw UserFriendlyException.SameMessage("当前节点状态已办理");
  1869. if (step.StepType is EStepType.End)
  1870. throw new UserFriendlyException("当前流程已流转到最终步骤");
  1871. if (dto.IsStartCountersign && !counterSignType.HasValue)
  1872. throw new UserFriendlyException("缺少会签类型参数");
  1873. //办理参数
  1874. //_mapper.Map(dto, step);
  1875. step.NextHandlers = dto.NextHandlers;
  1876. step.NextMainHandler = dto.NextMainHandler;
  1877. step.NextStepCode = dto.NextStepCode;
  1878. step.IsSms = dto.IsSms;
  1879. step.Opinion = dto.Opinion;
  1880. step.Remark = dto.Remark;
  1881. step.ReviewResult = dto.ReviewResult;
  1882. if (workflow.FlowType is EFlowType.Review)
  1883. step.HandleMode = dto.ReviewResult == EReviewResult.Approval
  1884. ? EHandleMode.Approved
  1885. : EHandleMode.NotApproved;
  1886. //step办理状态
  1887. HandleStep(step, handleMode, dto.Opinion, dto.NextStepCode);
  1888. }
  1889. /// <summary>
  1890. /// 查找会签循环的初始会签发起节点
  1891. /// </summary>
  1892. /// <param name="steps"></param>
  1893. /// <param name="currentStep"></param>
  1894. /// <returns></returns>
  1895. /// <exception cref="UserFriendlyException"></exception>
  1896. public WorkflowStep GetCsLoopStartStep(List<WorkflowStep> steps, WorkflowStep currentStep)
  1897. {
  1898. var startCountersignStep = steps.FirstOrDefault(d => d.Id == currentStep.CountersignStartStepId);
  1899. if (startCountersignStep is null)
  1900. throw new UserFriendlyException(
  1901. $"GetCsLoopStartStep: 未查询到会签开始节点,workflowId: {currentStep.WorkflowId}, startStepId: {currentStep.CountersignStartStepId}",
  1902. "未查询到会签开始节点,数据异常");
  1903. if (!startCountersignStep.IsCountersignEndStep)
  1904. return startCountersignStep;
  1905. return GetCsLoopStartStep(steps, startCountersignStep);
  1906. }
  1907. public async Task HandlePublishTraceAsync(string workflowId, string orderPublishId,
  1908. UserInfo acceptor, UserInfo handler, DateTime handleTime, UserInfo visitAcceptor, string orderVisitId, CancellationToken cancellation)
  1909. {
  1910. if (string.IsNullOrEmpty(orderPublishId))
  1911. throw new UserFriendlyException($"参数异常,orderPublishId不能为空, workflowId: {workflowId}");
  1912. //handle pubtrace
  1913. var pubTrace = await _workflowTraceRepository.Queryable()
  1914. .FirstAsync(d => d.WorkflowId == workflowId
  1915. && d.TraceStyle == ETraceStyle.Publish
  1916. && d.Status == EWorkflowStepStatus.WaitForAccept, cancellation);
  1917. if (pubTrace is not null)
  1918. {
  1919. pubTrace.OrderPublishId = orderPublishId;
  1920. pubTrace.AcceptorId = acceptor.UserId;
  1921. pubTrace.AcceptorName = acceptor.UserName;
  1922. pubTrace.AcceptorOrgId = acceptor.OrgId;
  1923. pubTrace.AcceptorOrgName = acceptor.OrgName;
  1924. pubTrace.AcceptTime = handleTime;
  1925. pubTrace.HandlerId = handler.UserId;
  1926. pubTrace.HandlerName = handler.UserName;
  1927. pubTrace.HandlerOrgId = handler.OrgId;
  1928. pubTrace.HandlerOrgName = handler.OrgName;
  1929. pubTrace.HandleTime = handleTime;
  1930. pubTrace.Status = EWorkflowStepStatus.Handled;
  1931. await _workflowTraceRepository.UpdateAsync(pubTrace, cancellation);
  1932. //create visit trace
  1933. await CreateVisitTraceAsync(pubTrace, visitAcceptor, orderVisitId, cancellation);
  1934. }
  1935. else
  1936. {
  1937. //throw new UserFriendlyException($"未查询到待办的发布节点, workflowId:{workflowId}");
  1938. }
  1939. }
  1940. public async Task HandleVisitTraceAsync(string orderVisitId, UserInfo visitor, DateTime visitTime, CancellationToken cancellation)
  1941. {
  1942. var visitTrace = await _workflowTraceRepository.GetAsync(d => d.OrderVisitId == orderVisitId,
  1943. cancellationToken: cancellation);
  1944. if (visitTrace is not null)
  1945. {
  1946. visitTrace.AcceptorId = visitor.UserId;
  1947. visitTrace.AcceptorName = visitor.UserName;
  1948. visitTrace.AcceptorOrgId = visitor.OrgId;
  1949. visitTrace.AcceptorOrgName = visitor.OrgName;
  1950. visitTrace.HandlerId = visitor.UserId;
  1951. visitTrace.HandlerName = visitor.UserName;
  1952. visitTrace.HandlerOrgId = visitor.OrgId;
  1953. visitTrace.HandlerOrgName = visitor.OrgName;
  1954. visitTrace.HandleTime = visitTime;
  1955. visitTrace.Status = EWorkflowStepStatus.Handled;
  1956. await _workflowTraceRepository.UpdateAsync(visitTrace, cancellation);
  1957. //create append end trace
  1958. await CreateTrashEndTraceAsync(visitTrace, cancellation);
  1959. }
  1960. else
  1961. {
  1962. //throw new UserFriendlyException($"未查询到待办的发布节点, orderVisitId:{orderVisitId}");
  1963. }
  1964. }
  1965. #region private method
  1966. private StepAssignInfo GetStepAssignInfo(ReverseFlowStepAssignInfo assignInfo,
  1967. WorkflowStep? targetStep = null, StepDefine? targetStepDefine = null)
  1968. {
  1969. switch (assignInfo.ReverseFlowStepCreationPolicy)
  1970. {
  1971. case EReverseFlowStepCreationPolicy.OriginStep:
  1972. if (targetStep is null)
  1973. throw new UserFriendlyException("参数异常:原节点信息为空");
  1974. return GetStepAssignInfo(targetStep);
  1975. case EReverseFlowStepCreationPolicy.OriginStepUser:
  1976. if (string.IsNullOrEmpty(targetStep?.HandlerId))
  1977. throw new UserFriendlyException("参数异常:原节点办理人为空");
  1978. return GetStepAssignInfo(targetStep, EFlowAssignType.User);
  1979. case EReverseFlowStepCreationPolicy.OriginStepOrg:
  1980. if (string.IsNullOrEmpty(targetStep?.HandlerOrgId))
  1981. throw new UserFriendlyException("参数异常:原节点办理部门为空");
  1982. return GetStepAssignInfo(targetStep, EFlowAssignType.Org);
  1983. case EReverseFlowStepCreationPolicy.OriginStepRole:
  1984. if (string.IsNullOrEmpty(targetStep?.RoleId))
  1985. throw new UserFriendlyException("参数异常:原节点办理角色为空");
  1986. return GetStepAssignInfo(targetStep, EFlowAssignType.Role);
  1987. case EReverseFlowStepCreationPolicy.OriginStepOrgAndRole:
  1988. if (string.IsNullOrEmpty(targetStep?.RoleId) || string.IsNullOrEmpty(targetStep?.HandlerOrgId))
  1989. throw new UserFriendlyException("参数异常:原节点办理角色或部门为空");
  1990. return GetStepAssignInfo(targetStep, EFlowAssignType.OrgAndRole);
  1991. case EReverseFlowStepCreationPolicy.AssignHandler:
  1992. if (assignInfo?.StepAssignInfo is null)
  1993. throw new UserFriendlyException("参数异常:节点指定办理对象信息为空");
  1994. return assignInfo.StepAssignInfo;
  1995. case EReverseFlowStepCreationPolicy.OriginDefinition:
  1996. if (targetStepDefine is null)
  1997. throw new UserFriendlyException("参数异常:节点配置信息为空");
  1998. return GetStepAssignInfo(targetStepDefine);
  1999. default:
  2000. throw new ArgumentOutOfRangeException();
  2001. }
  2002. }
  2003. private StepAssignInfo GetStepAssignInfo(StepDefine stepDefine)
  2004. {
  2005. var handler = stepDefine.HandlerTypeItems.FirstOrDefault();
  2006. if (handler is null)
  2007. throw new UserFriendlyException($"未正确配置节点办理对象, stepcode: {stepDefine.Code}", "未正确配置节点办理对象");
  2008. var rsp = new StepAssignInfo();
  2009. switch (stepDefine.HandlerType)
  2010. {
  2011. case EHandlerType.Role:
  2012. rsp.FlowAssignType = EFlowAssignType.Role;
  2013. rsp.RoleId = handler.Key;
  2014. rsp.RoleName = handler.Value;
  2015. break;
  2016. case EHandlerType.AssignedUser:
  2017. rsp.FlowAssignType = EFlowAssignType.User;
  2018. rsp.UserId = handler.Key;
  2019. rsp.Username = handler.Value;
  2020. break;
  2021. case EHandlerType.AssignedOrg:
  2022. rsp.FlowAssignType = EFlowAssignType.Org;
  2023. rsp.OrgId = handler.Key;
  2024. rsp.OrgName = handler.Value;
  2025. break;
  2026. case EHandlerType.OrgType:
  2027. throw new ArgumentOutOfRangeException("部门类型不支持按配置指派");
  2028. case EHandlerType.OrgLevel:
  2029. throw new ArgumentOutOfRangeException("部门等级不支持按配置指派");
  2030. default:
  2031. throw new ArgumentOutOfRangeException();
  2032. }
  2033. return rsp;
  2034. }
  2035. private StepAssignInfo GetStepAssignInfo(WorkflowStep targetStep, EFlowAssignType? flowAssignType = null)
  2036. {
  2037. return new StepAssignInfo
  2038. {
  2039. FlowAssignType = (flowAssignType ?? targetStep.FlowAssignType) ?? EFlowAssignType.User,
  2040. UserId = targetStep.HandlerId,
  2041. Username = targetStep.HandlerName,
  2042. OrgId = targetStep.HandlerOrgId,
  2043. OrgName = targetStep.HandlerOrgName,
  2044. RoleId = targetStep.RoleId,
  2045. RoleName = targetStep.RoleName,
  2046. };
  2047. }
  2048. private Workflow CreateWorkflow(WorkflowModule wfModule, string title, string? externalId = null)
  2049. {
  2050. var definition = wfModule.Definition;
  2051. if (definition is null)
  2052. throw new UserFriendlyException("无效流程模板");
  2053. var workflow = new Workflow
  2054. {
  2055. Title = title,
  2056. ModuleId = wfModule.Id,
  2057. ModuleName = wfModule.Name,
  2058. ModuleCode = wfModule.Code,
  2059. DefinitionId = definition.Id,
  2060. Status = EWorkflowStatus.Runnable,
  2061. Steps = new(),
  2062. Traces = new(),
  2063. WorkflowDefinition = definition,
  2064. ExternalId = externalId ?? string.Empty,
  2065. FlowType = definition.FlowType,
  2066. };
  2067. workflow.InitId();
  2068. return workflow;
  2069. }
  2070. private static void UpdateCurrentStep(Workflow workflow, BasicWorkflowDto dto,
  2071. StepDefine nextStepDefine, List<WorkflowStep> nextSteps)
  2072. {
  2073. if (dto.IsStartCountersign) return;
  2074. if (workflow.IsInCountersign) return;
  2075. if (nextStepDefine.BusinessType is EBusinessType.Seat or EBusinessType.Send)
  2076. {
  2077. //坐席->派单不选办理对象时
  2078. workflow.UpdateCurrentStepWhenAssign(nextSteps.First(),
  2079. new FlowStepHandler
  2080. {
  2081. OrgId = OrgSeedData.CenterId,
  2082. OrgName = OrgSeedData.CenterName
  2083. });
  2084. }
  2085. else
  2086. {
  2087. var nextHandler = dto.NextHandlers.First();
  2088. workflow.UpdateCurrentStepWhenAssign(nextSteps.First(), nextHandler);
  2089. }
  2090. }
  2091. private static void UpdateActualStep(Workflow workflow, BasicWorkflowDto dto,
  2092. StepDefine nextStepDefine, List<WorkflowStep> nextSteps)
  2093. {
  2094. if (dto.IsStartCountersign) return;
  2095. if (workflow.IsInCountersign) return;
  2096. if (nextStepDefine.StepType is EStepType.Summary or EStepType.End) return;
  2097. if (nextSteps.Count > 1) return; //多个下级节点不更新workflow的实际办理信息
  2098. var nextStep = nextSteps.First();
  2099. //todo 重构为办理对象由参数传入,指派给中心?派单员?待确认
  2100. if (nextStepDefine.BusinessType is EBusinessType.Seat or EBusinessType.Send)
  2101. {
  2102. //坐席->派单不选办理对象时
  2103. workflow.UpdateActualStepWhenAssign(nextStep,
  2104. new FlowStepHandler
  2105. {
  2106. Key = OrgSeedData.CenterId,
  2107. Value = OrgSeedData.CenterName,
  2108. OrgId = OrgSeedData.CenterId,
  2109. OrgName = OrgSeedData.CenterName
  2110. });
  2111. }
  2112. else
  2113. {
  2114. var nextHandler = dto.NextHandlers.First();
  2115. workflow.UpdateActualStepWhenAssign(nextStep, nextHandler);
  2116. }
  2117. //与实际办理节点的接办状态保持一致
  2118. workflow.ActualHandleStepAcceptTime = nextStep.AcceptTime;
  2119. }
  2120. private async Task<WorkflowStep> CreateStartStepAsync(Workflow workflow, StepDefine startStepDefine,
  2121. BasicWorkflowDto dto, UserInfo assigner, StepAssignInfo stepAssignInfo,
  2122. EWorkflowTraceType traceType, DateTime? expiredTime,
  2123. Action<Workflow, WorkflowStep, StepDefine, WorkflowStep>? stepConfig = null,
  2124. CancellationToken cancellationToken = default)
  2125. {
  2126. var startStep = CreateStartStep(workflow, startStepDefine, dto, assigner, stepAssignInfo, expiredTime, stepConfig);
  2127. await _workflowStepRepository.AddAsync(startStep, cancellationToken);
  2128. await CreateTraceAsync(workflow, startStep, traceType, cancellationToken);
  2129. return startStep;
  2130. }
  2131. /// <summary>
  2132. /// 创建下1/N个节点
  2133. /// </summary>
  2134. private async Task<List<WorkflowStep>> CreateNextStepsAsync(Workflow workflow, WorkflowStep currentStep,
  2135. BasicWorkflowDto dto, StepDefine nextStepDefine, UserInfo assigner, bool isNextDynamic,
  2136. DateTime? expiredTime, bool isStartCountersign, bool isAutoFillSummaryOpinion = false,
  2137. Action<Workflow, WorkflowStep, StepDefine, WorkflowStep>? newStepConfig = null,
  2138. CancellationToken cancellationToken = default)
  2139. {
  2140. List<WorkflowStep> nextSteps = new();
  2141. if (currentStep.IsInCountersign())
  2142. {
  2143. if (currentStep.IsCountersignEndStep)
  2144. {
  2145. // check if current is topend f: csStartStep.prev
  2146. // t: check if dto.StartCs t: csconfig f: config
  2147. if (currentStep.IsTopCountersignEndStep(workflow.TopCountersignStepId))
  2148. {
  2149. if (isStartCountersign)
  2150. {
  2151. //依据会签策略创建会签下一级节点
  2152. nextSteps = await CreateCountersignStepsAsync(workflow, nextStepDefine, currentStep, dto,
  2153. assigner, expiredTime, newStepConfig, cancellationToken);
  2154. }
  2155. else
  2156. {
  2157. //创建普通节点(根据配置)
  2158. nextSteps = await CreateConfigStepsAsync(workflow, nextStepDefine, currentStep, dto, assigner,
  2159. EWorkflowTraceType.Normal, expiredTime, isAutoFillSummaryOpinion, newStepConfig, cancellationToken);
  2160. }
  2161. }
  2162. else
  2163. {
  2164. if (dto.BackToCountersignEnd)
  2165. {
  2166. // csStartStep.prev
  2167. var csStartStep = GetRealCsStartHandleStep(workflow.Steps, currentStep.CountersignStartStepId);
  2168. if (csStartStep is null)
  2169. throw new UserFriendlyException("未查询到会签节点");
  2170. nextSteps = await CreateCsEndStepsByTargetPrevAsync(workflow, csStartStep, dto, expiredTime,
  2171. isAutoFillSummaryOpinion, cancellationToken);
  2172. }
  2173. else
  2174. {
  2175. //依据会签策略创建会签下一级节点
  2176. nextSteps = await CreateCountersignStepsAsync(workflow, nextStepDefine, currentStep, dto,
  2177. assigner, expiredTime, newStepConfig, cancellationToken);
  2178. }
  2179. }
  2180. }
  2181. else
  2182. {
  2183. if (dto.BackToCountersignEnd)
  2184. {
  2185. // check if cs all complete, create next
  2186. nextSteps = await CreateCsEndStepsByTargetPrevAsync(workflow, currentStep, dto,
  2187. expiredTime, isAutoFillSummaryOpinion, cancellationToken);
  2188. }
  2189. else
  2190. {
  2191. //依据会签策略创建会签下一级节点
  2192. nextSteps = await CreateCountersignStepsAsync(workflow, nextStepDefine, currentStep, dto,
  2193. assigner, expiredTime, newStepConfig, cancellationToken);
  2194. }
  2195. }
  2196. }
  2197. else if (isStartCountersign) //top
  2198. {
  2199. //依据会签策略创建会签下一级节点
  2200. nextSteps = await CreateCountersignStepsAsync(workflow, nextStepDefine, currentStep, dto,
  2201. assigner, expiredTime, newStepConfig, cancellationToken);
  2202. }
  2203. else if (isNextDynamic)
  2204. {
  2205. //创建动态下一级节点
  2206. nextSteps = await CreateDynamicStepsAsync(workflow, nextStepDefine, currentStep, dto, assigner,
  2207. expiredTime, newStepConfig, cancellationToken);
  2208. }
  2209. else
  2210. {
  2211. //创建普通节点(根据配置)
  2212. nextSteps = await CreateConfigStepsAsync(workflow, nextStepDefine, currentStep, dto, assigner,
  2213. EWorkflowTraceType.Normal, expiredTime, isAutoFillSummaryOpinion, newStepConfig, cancellationToken);
  2214. }
  2215. return nextSteps;
  2216. }
  2217. /// <summary>
  2218. /// 查询当前会签开始的第一级办理节点(会签汇总再次发起会签得场景)
  2219. /// </summary>
  2220. private static WorkflowStep? GetRealCsStartHandleStep(List<WorkflowStep> steps, string countersignStartStepId)
  2221. {
  2222. if (string.IsNullOrEmpty(countersignStartStepId))
  2223. throw new UserFriendlyException("会签汇总节点未正确赋值会签开始节点编号");
  2224. var csStartStep = steps.FirstOrDefault(d => d.Id == countersignStartStepId);
  2225. if (csStartStep.IsCountersignEndStep)
  2226. csStartStep = GetRealCsStartHandleStep(steps, csStartStep.CountersignStartStepId);
  2227. return csStartStep;
  2228. }
  2229. private async Task<List<WorkflowStep>> CreateDynamicStepsAsync(
  2230. Workflow workflow,
  2231. StepDefine nextStepDefine,
  2232. WorkflowStep currentStep,
  2233. BasicWorkflowDto dto,
  2234. UserInfo assigner,
  2235. DateTime? expiredTime,
  2236. Action<Workflow, WorkflowStep, StepDefine, WorkflowStep>? newStepConfig = null,
  2237. CancellationToken cancellationToken = default)
  2238. {
  2239. var handlerType = nextStepDefine.InstancePolicy switch
  2240. {
  2241. EDynamicPolicy.OrgUpCenterTop => EHandlerType.OrgLevel,
  2242. EDynamicPolicy.OrgUp => EHandlerType.OrgLevel,
  2243. EDynamicPolicy.OrgUpHandleCenterTop => EHandlerType.OrgLevel,
  2244. EDynamicPolicy.OrgUpHandle => EHandlerType.OrgLevel,
  2245. EDynamicPolicy.OrgUpLeadCenterTop => EHandlerType.OrgLevel,
  2246. EDynamicPolicy.OrgUpLead => EHandlerType.OrgLevel,
  2247. EDynamicPolicy.ArriveCenter => EHandlerType.OrgLevel,
  2248. EDynamicPolicy.ArriveOneOrg => EHandlerType.OrgLevel,
  2249. EDynamicPolicy.OrgDownCenterTop => EHandlerType.OrgLevel,
  2250. EDynamicPolicy.OrgDown => EHandlerType.OrgLevel,
  2251. null => throw new ArgumentOutOfRangeException(),
  2252. _ => throw new ArgumentOutOfRangeException()
  2253. };
  2254. return await CreateStepsAsync(workflow, nextStepDefine, currentStep, dto, assigner,
  2255. dto.NextHandlers, null, EWorkflowStepStatus.WaitForAccept,
  2256. ECountersignPosition.None, false, EWorkflowTraceType.Normal, handlerType, expiredTime,
  2257. newStepConfig: newStepConfig, cancellationToken: cancellationToken);
  2258. }
  2259. private Task<List<WorkflowStep>> CreateCountersignStepsAsync(
  2260. Workflow workflow,
  2261. StepDefine stepDefine,
  2262. WorkflowStep currentStep,
  2263. BasicWorkflowDto dto,
  2264. UserInfo assigner,
  2265. DateTime? expiredTime,
  2266. Action<Workflow, WorkflowStep, StepDefine, WorkflowStep>? newStepConfig = null,
  2267. CancellationToken cancellationToken = default
  2268. )
  2269. {
  2270. //var countersignId = dto.IsStartCountersign ? prevStep.StartCountersignId : prevStep.CountersignId;
  2271. var countersignId = currentStep.StartCountersignId;
  2272. //当前策略均为orglevel
  2273. var handlerType = EHandlerType.OrgLevel;
  2274. var nextStepCountersignPosition = dto.NextHandlers.Count > 1
  2275. ? ECountersignPosition.Direct
  2276. : ECountersignPosition.Indirect;
  2277. return CreateStepsAsync(workflow, stepDefine, currentStep, dto, assigner, dto.NextHandlers,
  2278. countersignId, EWorkflowStepStatus.WaitForAccept, nextStepCountersignPosition,
  2279. false, EWorkflowTraceType.Normal, handlerType, expiredTime, newStepConfig: newStepConfig, cancellationToken: cancellationToken);
  2280. }
  2281. /// <summary>
  2282. /// 根据传入节点的上一节点创建会签汇总节点(汇总传入节点的前一节点)
  2283. /// </summary>
  2284. private async Task<List<WorkflowStep>> CreateCsEndStepsByTargetPrevAsync(Workflow workflow, WorkflowStep step,
  2285. BasicWorkflowDto dto, DateTime? expiredTime, bool isAutoFillSummaryOpinion = false, CancellationToken cancellationToken = default)
  2286. {
  2287. //var countersignStartStep = workflow.Steps.FirstOrDefault(d => d.Id == step.PrevStepId);
  2288. var countersignStartStep = workflow.Steps.FirstOrDefault(d => d.StartCountersignId == step.CountersignId);
  2289. if (countersignStartStep is null)
  2290. throw new UserFriendlyException("未查询到当前节点上级会签开启节点");
  2291. var nextSteps = new List<WorkflowStep>();
  2292. //会签未全部办理则不创建汇总节点
  2293. //var csInnerSteps = workflow.Steps.Where(d => d.PrevStepId == countersignStartStep.Id).ToList();
  2294. //if (csInnerSteps.Any(d =>
  2295. // d.Status != EWorkflowStepStatus.Handled || (d.IsStartCountersign && !d.IsStartedCountersignEnd)))
  2296. if (!HasStepsAllHandled(workflow.Steps, countersignStartStep))
  2297. return nextSteps;
  2298. string? opinion = null;
  2299. if (isAutoFillSummaryOpinion)
  2300. {
  2301. //依据某节点作为根节点,查找最底层办理节点无论普通还是汇总节点
  2302. var preSteps = GetLastStepsFromRootStep(workflow.Steps, countersignStartStep);
  2303. var sb = new StringBuilder();
  2304. foreach (var prevStep in preSteps.OrderBy(d => d.HandleTime).ToList())
  2305. {
  2306. sb.AppendLine($"【会签时间】:{prevStep.HandleTime?.ToString("yyyy-MM-dd HH:mm:ss")}");
  2307. sb.AppendLine($"【会签人】:{prevStep.HandlerName}");
  2308. sb.AppendLine($"【会签结果】:{prevStep.Opinion}");
  2309. sb.AppendLine("");
  2310. }
  2311. opinion = sb.ToString();
  2312. }
  2313. // 创建会签汇总节点
  2314. var countersignEndStep =
  2315. await CreateCountersignEndStepAsync(workflow, countersignStartStep, dto, expiredTime, opinion, cancellationToken);
  2316. nextSteps = new List<WorkflowStep> { countersignEndStep };
  2317. //create trace
  2318. await CreateTraceAsync(workflow, countersignEndStep, EWorkflowTraceType.Normal, cancellationToken);
  2319. await _publisher.PublishAsync(new CountersignEndAssigned(workflow), PublishStrategy.ParallelWhenAll,
  2320. cancellationToken);
  2321. return nextSteps;
  2322. }
  2323. /// <summary>
  2324. /// 是否会签节点都已经办理完成(包含会签嵌套的判断)
  2325. /// </summary>
  2326. private bool HasStepsAllHandled(List<WorkflowStep> steps, WorkflowStep countersignStartStep)
  2327. {
  2328. var countersignSteps = GetStepsCascade(steps, countersignStartStep);
  2329. return countersignSteps.All(d => d.Status == EWorkflowStepStatus.Handled);
  2330. }
  2331. private List<WorkflowStep> GetStepsCascade(List<WorkflowStep> steps, WorkflowStep countersignStartStep)
  2332. {
  2333. _logger.LogWarning($"wfId: {countersignStartStep.WorkflowId}, stepId: {countersignStartStep.Id}");
  2334. var rsp = new List<WorkflowStep> { countersignStartStep };
  2335. if (!countersignStartStep.IsStartCountersign) return rsp;
  2336. var innerSteps = steps.Where(d => !string.IsNullOrEmpty(d.CountersignId)
  2337. && d.CountersignId == countersignStartStep.StartCountersignId).ToList();
  2338. _logger.LogWarning($"wfId: {countersignStartStep.WorkflowId}, innerSteps count: {innerSteps.Count}");
  2339. if (!innerSteps.Any()) return rsp;
  2340. foreach (var innerStep in innerSteps)
  2341. {
  2342. rsp.AddRange(GetStepsCascade(steps, innerStep));
  2343. }
  2344. return rsp;
  2345. }
  2346. /// <summary>
  2347. /// 以某一节点作为根节点开始查找最底层办理节点(需求:自动填充汇总节点意见)
  2348. /// </summary>
  2349. private List<WorkflowStep> GetLastStepsFromRootStep(List<WorkflowStep> steps, WorkflowStep rootStep)
  2350. {
  2351. var lastSteps = new List<WorkflowStep>();
  2352. var nextSteps = steps.Where(d => d.PrevStepId == rootStep.Id).ToList();
  2353. foreach (var nextStep in nextSteps)
  2354. {
  2355. if (nextStep.IsStartCountersign)
  2356. {
  2357. //find last csend
  2358. var lastCsEndStep = GetLastCountersignEndStep(steps, nextStep);
  2359. if (lastCsEndStep != null)
  2360. lastSteps.Add(lastCsEndStep);
  2361. }
  2362. else
  2363. {
  2364. lastSteps.Add(nextStep);
  2365. }
  2366. }
  2367. return lastSteps;
  2368. }
  2369. private WorkflowStep? GetLastCountersignEndStep(List<WorkflowStep> steps, WorkflowStep nextStep)
  2370. {
  2371. var csEndStep = steps.FirstOrDefault(d => d.IsCountersignEndStep && d.CountersignStartStepId == nextStep.Id);
  2372. if (csEndStep == null) return null;
  2373. return !csEndStep.IsStartCountersign ? csEndStep : GetLastCountersignEndStep(steps, csEndStep);
  2374. }
  2375. private async Task<WorkflowStep> CreateCountersignEndStepAsync(
  2376. Workflow workflow, WorkflowStep countersignStartStep,
  2377. BasicWorkflowDto dto, DateTime? expiredTime, string? opinion = null,
  2378. CancellationToken cancellationToken = default)
  2379. {
  2380. var csEndStep = _mapper.Map<WorkflowStep>(countersignStartStep);
  2381. csEndStep.Status = EWorkflowStepStatus.WaitForAccept;
  2382. csEndStep.PrevStepId = null;
  2383. csEndStep.PrevStepCode = null;
  2384. csEndStep.IsOrigin = false;
  2385. csEndStep.CountersignId = countersignStartStep.StartCountersignId;
  2386. csEndStep.CountersignPosition = ECountersignPosition.End;
  2387. //csEndStep.CountersignSteps = new();
  2388. csEndStep.IsCountersignEndStep = true;
  2389. csEndStep.CountersignStartStepId = countersignStartStep.Id;
  2390. csEndStep.Name = dto.NextStepName;
  2391. //csEndStep.TimeLimit = GetTimeLimit("");
  2392. csEndStep.StepExpiredTime = expiredTime;
  2393. csEndStep.BusinessType = dto.BusinessType;
  2394. csEndStep.Handlers = countersignStartStep.Handlers
  2395. .Where(d => d.Key == countersignStartStep.HandlerId || d.Key == countersignStartStep.HandlerOrgId)
  2396. .ToList();
  2397. //需求调整:汇总节点指派给发起人部门办理 //todo 待重构
  2398. csEndStep.FlowAssignType = EFlowAssignType.Org;
  2399. csEndStep.Reset();
  2400. csEndStep.ResetParameters();
  2401. if (!string.IsNullOrEmpty(opinion))
  2402. csEndStep.Opinion = opinion;
  2403. await _workflowStepRepository.AddAsync(csEndStep, cancellationToken);
  2404. workflow.Steps.Add(csEndStep);
  2405. //await _workflowStepRepository.AddNav(csEndStep)
  2406. // .Include(d => d.StepHandlers)
  2407. // .ExecuteCommandAsync();
  2408. return csEndStep;
  2409. }
  2410. private bool CheckIsActualHandle(Workflow workflow, WorkflowStep step, StepDefine nextStepDefine,
  2411. BasicWorkflowDto dto)
  2412. {
  2413. //1. workflow是否为办理类型 2. 非会签:当前是否为普通节点and下一节点是否为汇总 or endStep
  2414. //3. 会签:当前操作为汇总还是继续往下办理?thk: 汇总以后但未回到top又往下办理的场景,前面实际办理部门也算作办理部门
  2415. if (workflow.FlowType is not EFlowType.Handle) return false;
  2416. if (workflow.IsInCountersign)
  2417. {
  2418. return !step.IsCountersignEndStep && dto.BackToCountersignEnd;
  2419. }
  2420. else
  2421. {
  2422. return step.StepType is EStepType.Normal &&
  2423. nextStepDefine.StepType is EStepType.Summary or EStepType.End;
  2424. }
  2425. }
  2426. /// <summary>
  2427. /// 办理节点(赋值节点的办理对象信息)
  2428. /// </summary>
  2429. private void HandleStep(WorkflowStep step, EHandleMode handleMode, string opinion, string nextStepCode)
  2430. {
  2431. //todo 重构:ISessionContext传入
  2432. step.Handle(_sessionContext.RequiredUserId, _sessionContext.UserName,
  2433. _sessionContext.RequiredOrgId, _sessionContext.OrgName,
  2434. _sessionContext.OrgAreaCode, _sessionContext.OrgAreaName,
  2435. _sessionContext.OrgIsCenter, handleMode, opinion, nextStepCode);
  2436. //var handler = step.FindActualHandler(current.Roles, current.RequiredUserId, current.RequiredOrgId);
  2437. //if (handler is not null)
  2438. // handler.IsActualHandler = true;
  2439. }
  2440. /// <summary>
  2441. /// 开始会签(创建会签数据,更新currentStep会签数据)
  2442. /// </summary>
  2443. private async Task<WorkflowCountersign> StartCountersignAsync(ISessionContext current, Workflow workflow, WorkflowStep startStep,
  2444. BasicWorkflowDto dto, ECounterSignType? counterSignType, DateTime? expiredTime,
  2445. CancellationToken cancellationToken)
  2446. {
  2447. var countersignId = startStep.CountersignId;
  2448. if (startStep.IsCountersignEndStep)
  2449. {
  2450. var topStartCsStep = GetCsLoopStartStep(workflow.Steps, startStep);
  2451. countersignId = topStartCsStep.CountersignId;
  2452. }
  2453. var countersign = await CreateCountersignAsync(current, workflow, startStep,
  2454. dto.NextHandlers.Select(d => new Kv(d.Key, d.Value)).ToList(), dto.FlowAssignType,
  2455. counterSignType, expiredTime, countersignId, cancellationToken);
  2456. startStep.StartCountersign(countersign.Id);
  2457. return countersign;
  2458. }
  2459. /// <summary>
  2460. /// 检查是否从中心流转至部门
  2461. /// </summary>
  2462. private bool CheckIfFlowFromCenterToOrg(WorkflowStep sourceStep, StepDefine targetStepBoxDefine)
  2463. {
  2464. var isFromCenter = sourceStep.IsCenter();
  2465. if (!isFromCenter) return false;
  2466. var isToOrg = targetStepBoxDefine.IsOrg();
  2467. return isFromCenter && isToOrg;
  2468. }
  2469. /// <summary>
  2470. /// 检查是否从中心流转至部门
  2471. /// </summary>
  2472. private bool CheckIfFlowFromCenterToOrg(Workflow workflow, WorkflowStep targetStepBox)
  2473. {
  2474. var isToOrg = targetStepBox.IsOrg();
  2475. if (!isToOrg) return false;
  2476. var isFromCenter = workflow.Steps.All(d => d.BusinessType is not EBusinessType.Department);
  2477. return isFromCenter && isToOrg;
  2478. }
  2479. /// <summary>
  2480. /// 检查是否从部门流转至中心
  2481. /// </summary>
  2482. private bool CheckIfFlowFromOrgToCenter(WorkflowStep sourceStepBox, StepDefine targetStepBoxDefine)
  2483. {
  2484. var isFromOrg = sourceStepBox.IsOrg();
  2485. if (!isFromOrg) return false;
  2486. var isToCenter = targetStepBoxDefine.IsCenter();
  2487. return isFromOrg && isToCenter;
  2488. }
  2489. /// <summary>
  2490. /// 检查是否从部门流转至中心
  2491. /// </summary>
  2492. private bool CheckIfFlowFromOrgToCenter(WorkflowStep sourceStepBox, WorkflowStep targetStep)
  2493. {
  2494. var isFromOrg = sourceStepBox.IsOrg();
  2495. if (!isFromOrg) return false;
  2496. var isToCenter = targetStep.IsCenter();
  2497. return isFromOrg && isToCenter;
  2498. }
  2499. /// <summary>
  2500. /// 检查是否从部门流转至中心
  2501. /// </summary>
  2502. private bool CheckIfFlowFromOrgToCenter(Workflow workflow, WorkflowStep targetStep)
  2503. {
  2504. var isToCenter = targetStep.IsCenter();
  2505. if (!isToCenter) return false;
  2506. var isFromOrg = workflow.Steps.Any(d => d.BusinessType is EBusinessType.Department or EBusinessType.DepartmentLeader);
  2507. return isFromOrg && isToCenter;
  2508. }
  2509. /// <summary>
  2510. /// 复制一个节点为待接办
  2511. /// </summary>
  2512. private async Task<WorkflowStep> DuplicateStepWithTraceAsync(Workflow workflow, StepDefine stepDefine, WorkflowStep step,
  2513. EWorkflowTraceType traceType, DateTime expiredTime, CancellationToken cancellationToken)
  2514. {
  2515. var newStep = DuplicateStep(workflow, stepDefine, step, traceType, expiredTime);
  2516. await _workflowStepRepository.AddAsync(newStep, cancellationToken);
  2517. await CreateTraceAsync(workflow, newStep, traceType, cancellationToken);
  2518. return newStep;
  2519. }
  2520. //private WorkflowStep DuplicateStep(WorkflowStep step, EWorkflowTraceType traceType, DateTime? expiredTime)
  2521. //{
  2522. // var newStep = _mapper.Map<WorkflowStep>(step);
  2523. // newStep.Reset();
  2524. // newStep.Status = EWorkflowStepStatus.WaitForAccept;
  2525. // newStep.PrevStepId = step.PrevStepId;
  2526. // newStep.IsMain = step.IsMain;
  2527. // newStep.IsOrigin = step.IsOrigin;
  2528. // //newStep.ParentId = step.ParentId;
  2529. // newStep.Handlers = step.Handlers;
  2530. // newStep.StartCountersignId = step.StartCountersignId;
  2531. // newStep.CountersignId = step.CountersignId;
  2532. // newStep.IsStartedCountersignEnd = step.IsStartedCountersignEnd;
  2533. // newStep.StepExpiredTime = expiredTime;
  2534. // newStep.InitId();
  2535. // newStep.Assign(step);
  2536. // //退回场景:指派给原办理人,其余场景:按照原节点原始指派方式复制 //todo 重构为参数传入办理对象
  2537. // if (traceType is EWorkflowTraceType.Previous)
  2538. // {
  2539. // newStep.FlowAssignType = step.FlowAssignType;
  2540. // // //newStep.FlowAssignType = EFlowAssignType.User;
  2541. // // // 是否中心 临时紧急修改 后续在流程模版定义是否原办理人退回类型 来实现流程 禅道200
  2542. // // //newStep.FlowAssignType = step.BusinessType is EBusinessType.Seat or EBusinessType.Send
  2543. // // // ? step.BusinessType is EBusinessType.Send ? EFlowAssignType.User : EFlowAssignType.Role
  2544. // // // : EFlowAssignType.Org;
  2545. // // //if (newStep is { FlowAssignType: EFlowAssignType.Role, BusinessType: EBusinessType.Send })
  2546. // // // newStep.FlowAssignType = EFlowAssignType.User;
  2547. // newStep.Assign(step.HandlerId, step.HandlerName, step.HandlerOrgId, step.HandlerOrgName, step.RoleId, step.RoleName);
  2548. // }
  2549. // return newStep;
  2550. //}
  2551. //new
  2552. private WorkflowStep DuplicateStep(
  2553. Workflow workflow, StepDefine stepDefine, WorkflowStep step,
  2554. EWorkflowTraceType traceType, DateTime? expiredTime)
  2555. {
  2556. var newStep = _mapper.Map<WorkflowStep>(step);
  2557. newStep.Reset();
  2558. newStep.Status = EWorkflowStepStatus.WaitForAccept;
  2559. newStep.PrevStepId = step.PrevStepId;
  2560. newStep.IsMain = step.IsMain;
  2561. newStep.IsOrigin = step.IsOrigin;
  2562. //newStep.ParentId = step.ParentId;
  2563. newStep.Handlers = step.Handlers;
  2564. newStep.StartCountersignId = step.StartCountersignId;
  2565. newStep.CountersignId = step.CountersignId;
  2566. newStep.IsStartedCountersignEnd = step.IsStartedCountersignEnd;
  2567. newStep.StepExpiredTime = expiredTime;
  2568. newStep.InitId();
  2569. newStep.Assign(step);
  2570. // //退回场景:指派给原办理人,其余场景:按照原节点原始指派方式复制 //todo 重构为参数传入办理对象
  2571. // if (traceType is EWorkflowTraceType.Previous)
  2572. // {
  2573. // newStep.FlowAssignType = step.FlowAssignType;
  2574. // // //newStep.FlowAssignType = EFlowAssignType.User;
  2575. // // // 是否中心 临时紧急修改 后续在流程模版定义是否原办理人退回类型 来实现流程 禅道200
  2576. // // //newStep.FlowAssignType = step.BusinessType is EBusinessType.Seat or EBusinessType.Send
  2577. // // // ? step.BusinessType is EBusinessType.Send ? EFlowAssignType.User : EFlowAssignType.Role
  2578. // // // : EFlowAssignType.Org;
  2579. // // //if (newStep is { FlowAssignType: EFlowAssignType.Role, BusinessType: EBusinessType.Send })
  2580. // // // newStep.FlowAssignType = EFlowAssignType.User;
  2581. //
  2582. // newStep.Assign(step.HandlerId, step.HandlerName, step.HandlerOrgId, step.HandlerOrgName, step.RoleId, step.RoleName);
  2583. // }
  2584. return newStep;
  2585. }
  2586. private async Task<WorkflowCountersign> CreateCountersignAsync(
  2587. ISessionContext current, Workflow workflow, WorkflowStep startStep, List<Kv> handlers,
  2588. EFlowAssignType? flowAssignType, ECounterSignType? counterSignType, DateTime? expiredTime,
  2589. string? parentId = null, CancellationToken cancellationToken = default)
  2590. {
  2591. var members = handlers.Select(d => new WorkflowCountersignMember
  2592. {
  2593. Key = d.Key,
  2594. Value = d.Value,
  2595. FlowAssignType = flowAssignType
  2596. }).ToList();
  2597. var countersign = new WorkflowCountersign
  2598. {
  2599. WorkflowId = workflow.Id,
  2600. StartStepId = startStep.Id,
  2601. StartStepCode = startStep.Code,
  2602. StartStepBusiType = startStep.BusinessType,
  2603. StarterId = current.UserId,
  2604. StarterName = current.UserName ?? string.Empty,
  2605. StarterOrgId = current.OrgId,
  2606. StarterOrgName = current.OrgName,
  2607. StarterOrgAreaCode = current.OrgAreaCode ?? string.Empty,
  2608. StarterOrgAreaName = current.OrgAreaName ?? string.Empty,
  2609. ParentId = parentId,
  2610. Members = members,
  2611. FlowAssignType = flowAssignType,
  2612. CounterSignType = counterSignType,
  2613. ExpiredTime = expiredTime,
  2614. //ExternalId = workflow.ExternalId,
  2615. };
  2616. //await _workflowCountersignRepository.AddAsync(countersign, cancellationToken);
  2617. await _workflowCountersignRepository.AddNav(countersign)
  2618. .Include(d => d.Members)
  2619. .ExecuteCommandAsync();
  2620. return countersign;
  2621. }
  2622. //private async Task JumpTraceAsync(string workflowId, RecallDto dto, CancellationToken cancellationToken)
  2623. //{
  2624. // //未办理的traces
  2625. // var uncompleteTraces =
  2626. // await _workflowTraceRepository.QueryAsync(d =>
  2627. // d.WorkflowId == workflowId && string.IsNullOrEmpty(d.HandlerId));
  2628. // foreach (var trace in uncompleteTraces)
  2629. // {
  2630. // HandleTrace(trace, dto.Opinion);
  2631. // }
  2632. // await _workflowTraceRepository.UpdateRangeAsync(uncompleteTraces, cancellationToken);
  2633. //}
  2634. //private async Task RecallTraceAsync(List<WorkflowTrace> traces, string opinion, ISessionContext current,
  2635. // CancellationToken cancellationToken)
  2636. //{
  2637. // //未办理的traces
  2638. // //var uncompleteTraces =
  2639. // // await _workflowTraceRepository.QueryAsync(d =>
  2640. // // d.WorkflowId == workflowId && string.IsNullOrEmpty(d.HandlerId));
  2641. // var uncompleteTraces = traces.Where(d => d.Status != EWorkflowStepStatus.Handled).ToList();
  2642. // if (uncompleteTraces.Any())
  2643. // {
  2644. // foreach (var trace in uncompleteTraces)
  2645. // {
  2646. // trace.Handle(
  2647. // current.RequiredUserId, current.UserName,
  2648. // current.RequiredOrgId, current.OrgName,
  2649. // current.OrgAreaCode, current.OrgAreaName,
  2650. // current.OrgIsCenter, opinion);
  2651. // }
  2652. // await _workflowTraceRepository.UpdateRangeAsync(uncompleteTraces, cancellationToken);
  2653. // }
  2654. //}
  2655. //private async Task<WorkflowTrace> PreviousTraceAsync(string workflowId,
  2656. // PreviousWorkflowDto dto, WorkflowStep step,
  2657. // string applicantId, string applicantName,
  2658. // string applicantOrgId, string applicantOrgName,
  2659. // string applicantOrgAreaCode, string applicantOrgAreaName,
  2660. // bool applicantIsCenter, CancellationToken cancellationToken)
  2661. //{
  2662. // var trace = await GetWorkflowTraceAsync(workflowId, step.Id, cancellationToken);
  2663. // _mapper.Map(dto, trace);
  2664. // //HandleTrace(trace, dto.Opinion, current);
  2665. // trace.Handle(applicantId, applicantName,
  2666. // applicantOrgId, applicantOrgName,
  2667. // applicantOrgAreaCode, applicantOrgAreaName,
  2668. // applicantIsCenter, dto.Opinion);
  2669. // await _workflowTraceRepository.UpdateAsync(trace, cancellationToken);
  2670. // return trace;
  2671. //}
  2672. //private async Task EndTraceAsync(Workflow workflow, BasicWorkflowDto dto, WorkflowStep step, CancellationToken cancellationToken)
  2673. //{
  2674. // var trace = _mapper.Map<WorkflowTrace>(step);
  2675. // trace.Status = EWorkflowTraceStatus.Normal;
  2676. // trace.ExpiredTime = workflow.ExpiredTime;
  2677. // trace.TimeLimit = workflow.TimeLimit;
  2678. // await _workflowTraceRepository.AddAsync(trace, cancellationToken);
  2679. //}
  2680. private async Task<WorkflowTrace> NextTraceAsync(Workflow workflow, BasicWorkflowDto dto, WorkflowStep step,
  2681. CancellationToken cancellationToken)
  2682. {
  2683. // var trace = await GetWorkflowTraceAsync(workflow.Id, step.Id, cancellationToken);
  2684. var trace = workflow.Traces.FirstOrDefault(d => d.Id == step.Id);
  2685. if (trace == null)
  2686. throw new UserFriendlyException($"未找到对应trace, workflowId: {workflow.Id}, stepId: {step.Id}");
  2687. _mapper.Map(dto, trace);
  2688. _mapper.Map(step, trace);
  2689. await _workflowTraceRepository.UpdateAsync(trace, cancellationToken);
  2690. return trace;
  2691. }
  2692. //private async Task AcceptTraceAsync(Workflow workflow, WorkflowStep step, CancellationToken cancellationToken)
  2693. //{
  2694. // var trace = await GetWorkflowTraceAsync(workflow.Id, step.Id, cancellationToken);
  2695. // _mapper.Map(step, trace);
  2696. // await _workflowTraceRepository.UpdateAsync(trace, cancellationToken);
  2697. //}
  2698. private async Task<WorkflowTrace> CreateTraceAsync(Workflow workflow, WorkflowStep step,
  2699. EWorkflowTraceType traceType = EWorkflowTraceType.Normal,
  2700. CancellationToken cancellationToken = default)
  2701. {
  2702. var sendHandleTimes = 0;
  2703. if (step.BusinessType == EBusinessType.Send)
  2704. {
  2705. var sendHandleCount = workflow.Traces.Count(d => d.StepType == EStepType.Normal &&
  2706. d.BusinessType == EBusinessType.Send);
  2707. sendHandleTimes = sendHandleCount + 1;
  2708. }
  2709. var trace = _mapper.Map<WorkflowTrace>(step);
  2710. trace.TraceType = traceType;
  2711. trace.SendHandleTimes = sendHandleTimes;
  2712. if (workflow.IsInCountersign && step.IsInCountersign())
  2713. {
  2714. if (step.IsCountersignEndStep)
  2715. {
  2716. //var startTrace = await GetWorkflowTraceAsync(workflow.Id, step.CountersignStartStepId, cancellationToken);
  2717. var startTrace = workflow.Traces.FirstOrDefault(d => d.Id == step.CountersignStartStepId);
  2718. if (startTrace == null)
  2719. throw new UserFriendlyException(
  2720. $"未找到startTrace, workflowId: {workflow.Id}, step.CountersignStartStepId: {step.CountersignStartStepId}");
  2721. trace.ParentId = startTrace.ParentId;
  2722. }
  2723. else
  2724. {
  2725. //if (step.CountersignPosition is ECountersignPosition.Direct)
  2726. //{
  2727. // var prevTrace = await GetWorkflowTraceAsync(workflow.Id, step.PrevStepId, cancellationToken);
  2728. // trace.ParentId = prevTrace.Id;
  2729. //}
  2730. //else if (step.CountersignPosition is ECountersignPosition.Single)
  2731. //{
  2732. // var prevTrace = await GetWorkflowTraceAsync(workflow.Id, step.PrevStepId, cancellationToken);
  2733. // trace.ParentId = prevTrace.ParentId;
  2734. //}
  2735. //var prevTrace = await GetWorkflowTraceAsync(workflow.Id, step.PrevStepId, cancellationToken);
  2736. var prevTrace = workflow.Traces.FirstOrDefault(d => d.Id == step.PrevStepId);
  2737. if (prevTrace == null)
  2738. throw new UserFriendlyException($"未找到prevTrace, workflowId: {workflow.Id}, step.PrevStepId: {step.PrevStepId}");
  2739. trace.ParentId = prevTrace.Id;
  2740. }
  2741. }
  2742. await _workflowTraceRepository.AddAsync(trace, cancellationToken);
  2743. workflow.Traces.Add(trace);
  2744. return trace;
  2745. }
  2746. private async Task<WorkflowTrace> GetWorkflowTraceAsync(string workflowId, string stepId,
  2747. CancellationToken cancellationToken)
  2748. {
  2749. var parentTrace = await _workflowTraceRepository.GetAsync(d =>
  2750. d.WorkflowId == workflowId && d.StepId == stepId, cancellationToken);
  2751. if (parentTrace == null)
  2752. throw new UserFriendlyException($"未找到对应trace, workflowId: {workflowId}, stepId: {stepId}");
  2753. return parentTrace;
  2754. }
  2755. // private async Task<bool> RecallAsync(Workflow workflow, BasicWorkflowDto dto, FlowAssignInfo flowAssignInfo,
  2756. // StepDefine targetStepDefine, WorkflowStep targetStep, EWorkflowTraceType traceType,
  2757. // DateTime? expiredTime, bool isOrderFiled, EHandleMode handleMode, CancellationToken cancellationToken)
  2758. // {
  2759. // var targetIsStartStep = targetStepDefine.StepType is EStepType.Start;
  2760. // var updateTraces = new List<WorkflowTrace>();
  2761. //
  2762. // //update uncomplete traces
  2763. // var uncompleteTraces = workflow.Traces.Where(d => d.Status != EWorkflowStepStatus.Handled && d.TraceStyle == ETraceStyle.Flow).ToList();
  2764. // if (uncompleteTraces.Any())
  2765. // {
  2766. // foreach (var trace in uncompleteTraces)
  2767. // {
  2768. // trace.Handle(
  2769. // _sessionContext.RequiredUserId, _sessionContext.UserName,
  2770. // _sessionContext.RequiredOrgId, _sessionContext.OrgName,
  2771. // _sessionContext.OrgAreaCode, _sessionContext.OrgAreaName,
  2772. // _sessionContext.OrgIsCenter, handleMode, dto.Opinion);
  2773. // }
  2774. //
  2775. // //await _workflowTraceRepository.UpdateRangeAsync(uncompleteTraces, cancellationToken);
  2776. // updateTraces.AddRange(uncompleteTraces);
  2777. // }
  2778. // else
  2779. // {
  2780. // var endTrace = workflow.Traces.Where(d => d.StepType == EStepType.End && d.TraceStyle == ETraceStyle.Flow).MaxBy(d => d.CreationTime);
  2781. // if (endTrace is not null)
  2782. // {
  2783. // endTrace.Opinion += ("\r\n" + dto.Opinion);
  2784. // updateTraces.Add(endTrace);
  2785. // }
  2786. // }
  2787. //
  2788. // //get targetStep's previous
  2789. // WorkflowStep? targetPrevStep = null;
  2790. // if (!targetIsStartStep)
  2791. // {
  2792. // targetPrevStep = workflow.Steps.FirstOrDefault(d => d.Id == targetStep.PrevStepId);
  2793. // if (targetPrevStep == null)
  2794. // throw new UserFriendlyException($"{nameof(RecallAsync)}, 未找到目标节点的前一节点, flowId: {workflow.Id}");
  2795. // }
  2796. //
  2797. // //查询所有目标节点之后的节点,然后删掉(包括目标节点)
  2798. // var removeSteps = GetStepsBehindTargetStep(workflow.Steps, targetStep);
  2799. // if (removeSteps.Any())
  2800. // {
  2801. // await _workflowStepRepository.RemoveRangeAsync(removeSteps, cancellationToken);
  2802. // workflow.Steps.RemoveAll(d => removeSteps.Contains(d));
  2803. //
  2804. // //更新快照对应节点状态
  2805. // var stepIds = removeSteps.Select(d => d.Id).ToList();
  2806. // var traces = workflow.Traces.Where(d => stepIds.Contains(d.StepId)).ToList();
  2807. // //await UpdateTracesStateAsync(updateTraces, EWorkflowTraceState.StepRemoveByRecall, cancellationToken);
  2808. // foreach (var trace in traces)
  2809. // {
  2810. // trace.TraceState = isOrderFiled
  2811. // ? EWorkflowTraceState.StepRemoveByRecallWhenFiled
  2812. // : EWorkflowTraceState.StepRemoveByRecall;
  2813. // }
  2814. //
  2815. // updateTraces.AddRange(traces);
  2816. // }
  2817. //
  2818. // await _workflowTraceRepository.UpdateRangeAsync(updateTraces, cancellationToken);
  2819. //
  2820. // //结束会签
  2821. // var unCompleteCountersigns = workflow.Countersigns.Where(d => !d.IsCompleted()).ToList();
  2822. // if (unCompleteCountersigns.Any())
  2823. // {
  2824. // foreach (var unCompleteCountersign in unCompleteCountersigns)
  2825. // {
  2826. // unCompleteCountersign.End(null, null, EBusinessType.File,
  2827. // _sessionContext.RequiredUserId, _sessionContext.UserName,
  2828. // _sessionContext.RequiredOrgId, _sessionContext.OrgName,
  2829. // _sessionContext.OrgAreaCode, _sessionContext.OrgAreaName);
  2830. // }
  2831. //
  2832. // await _workflowCountersignRepository.UpdateRangeAsync(unCompleteCountersigns, cancellationToken);
  2833. // }
  2834. //
  2835. // workflow.EndCountersign();
  2836. // workflow.ResetOption();
  2837. // if (workflow.Status is EWorkflowStatus.Completed)
  2838. // workflow.SetStatusRunnable();
  2839. //
  2840. // var assigner = new UserInfo(
  2841. // _sessionContext.UserId,
  2842. // _sessionContext.UserName,
  2843. // _sessionContext.OrgId,
  2844. // _sessionContext.OrgName,
  2845. // _sessionContext.OrgIsCenter
  2846. // );
  2847. //
  2848. // var targetStepNew = targetIsStartStep
  2849. // ? await CreateStartStepAsync(workflow, targetStepDefine, dto, assigner,
  2850. // dto.NextHandlers.First(), traceType, expiredTime, flowAssignInfo.FlowAssignType, cancellationToken)
  2851. // : (await CreateStepsAsync(workflow, targetStepDefine, targetPrevStep, dto, assigner, dto.NextHandlers,
  2852. // null, EWorkflowStepStatus.WaitForAccept, ECountersignPosition.None, true, traceType,
  2853. // null, expiredTime, cancellationToken: cancellationToken)).First();
  2854. //
  2855. // //更新实际办理节点信息
  2856. // workflow.UpdateActualStepWhenAssign(targetStepNew, new FlowStepHandler
  2857. // {
  2858. // UserId = targetStep.HandlerId,
  2859. // Username = targetStep.HandlerName,
  2860. // OrgId = targetStep.HandlerOrgId,
  2861. // OrgName = targetStep.HandlerOrgName
  2862. // });
  2863. //
  2864. // workflow.UpdateCurrentStepWhenAssign(targetStepNew, new FlowStepHandler
  2865. // {
  2866. // UserId = targetStep.HandlerId,
  2867. // Username = targetStep.HandlerName,
  2868. // OrgId = targetStep.HandlerOrgId,
  2869. // OrgName = targetStep.HandlerOrgName
  2870. // });
  2871. //
  2872. // //取消维护workflow得冗余字段(FlowedOrgIds, HandlerOrgs)
  2873. // //workflow.Assign(flowAssignInfo.FlowAssignType, flowAssignInfo.GetHandlerIds());
  2874. // //workflow.ResetHandlers(flowAssignInfo.FlowAssignType, flowAssignInfo.HandlerObjects);
  2875. //
  2876. // //calc workflow expired time
  2877. // var isOrgToCenter = CheckIfFlowFromOrgToCenter(workflow, targetStep);
  2878. //
  2879. // return isOrgToCenter;
  2880. // }
  2881. private List<WorkflowStep> GetStepsBehindTargetStep(List<WorkflowStep> steps, WorkflowStep targetStep)
  2882. {
  2883. var behindSteps = new List<WorkflowStep> { targetStep };
  2884. if (!steps.Any()) return behindSteps;
  2885. var nextSteps = targetStep.IsStartCountersign
  2886. ? steps.Where(d => d.CountersignId == targetStep.StartCountersignId).ToList()
  2887. : steps.Where(d => d.PrevStepId == targetStep.Id).ToList();
  2888. //var nextSteps = steps.Where(d => d.PrevStepId == targetStep.Id).ToList();
  2889. if (!nextSteps.Any())
  2890. return behindSteps;
  2891. foreach (var nextStep in nextSteps)
  2892. {
  2893. var leftSteps = steps.Except(behindSteps).ToList();
  2894. behindSteps.AddRange(GetStepsBehindTargetStep(leftSteps, nextStep));
  2895. }
  2896. return behindSteps;
  2897. }
  2898. private static void CheckWhetherRunnable(EWorkflowStatus status)
  2899. {
  2900. if (status != EWorkflowStatus.Runnable)
  2901. throw UserFriendlyException.SameMessage("当前流程状态不可继续流转");
  2902. }
  2903. private void ValidatePermission(Workflow workflow, string OrgId, string UserId, string[] roleIds)
  2904. {
  2905. if (!workflow.IsCanHandle(UserId, OrgId, roleIds))
  2906. throw UserFriendlyException.SameMessage("无办理权限");
  2907. }
  2908. private async Task<WorkflowStep> CreateEndStepAsync(
  2909. Workflow workflow,
  2910. StepDefine endStepDefine,
  2911. WorkflowStep prevStep,
  2912. BasicWorkflowDto dto,
  2913. DateTime? expiredTime,
  2914. CancellationToken cancellationToken)
  2915. {
  2916. if (workflow.Steps.Any(d => d.StepType == EStepType.End))
  2917. throw UserFriendlyException.SameMessage("无法重复创建结束节点");
  2918. var handler = new StepAssignInfo
  2919. {
  2920. Key = _sessionContext.UserId,
  2921. Value = _sessionContext.UserName,
  2922. UserId = _sessionContext.RequiredUserId,
  2923. Username = _sessionContext.UserName,
  2924. OrgId = _sessionContext.OrgId,
  2925. OrgName = _sessionContext.OrgName,
  2926. FlowAssignType = EFlowAssignType.User,
  2927. };
  2928. var assigner = new UserInfo(
  2929. _sessionContext.UserId,
  2930. _sessionContext.UserName,
  2931. _sessionContext.OrgId,
  2932. _sessionContext.OrgName,
  2933. _sessionContext.OrgIsCenter
  2934. );
  2935. var step = CreateStep(workflow, endStepDefine, prevStep, handler, assigner,
  2936. null, null, EWorkflowStepStatus.WaitForAccept,
  2937. ECountersignPosition.None, expiredTime, endStepDefine.Name, true, businessType: EBusinessType.File, flowDirection: dto.FlowDirection);
  2938. //step.Accept(_sessionContext.RequiredUserId, _sessionContext.UserName,
  2939. // _sessionContext.RequiredOrgId, _sessionContext.OrgName,
  2940. // _sessionContext.OrgAreaCode, _sessionContext.OrgAreaName);
  2941. var opinion = workflow.FlowType == EFlowType.Handle
  2942. ? $"流程归档。承办意见:{dto.Opinion}"
  2943. : "流程归档";
  2944. HandleStep(step, EHandleMode.Normal, opinion, string.Empty);
  2945. await _workflowStepRepository.AddAsync(step, cancellationToken);
  2946. workflow.Steps.Add(step);
  2947. //end trace
  2948. await CreateTraceAsync(workflow, step, cancellationToken: cancellationToken);
  2949. return step;
  2950. }
  2951. public async Task<List<WorkflowStep>> CreateConfigStepsAsync(
  2952. Workflow workflow,
  2953. StepDefine stepDefine,
  2954. WorkflowStep currentStep,
  2955. BasicWorkflowDto dto,
  2956. UserInfo assigner,
  2957. EWorkflowTraceType traceType,
  2958. DateTime? expiredTime,
  2959. bool isAutoFillSummaryOpinion = false,
  2960. Action<Workflow, WorkflowStep, StepDefine, WorkflowStep>? newStepConfig = null,
  2961. CancellationToken cancellationToken = default)
  2962. {
  2963. string? opinion = null;
  2964. if (isAutoFillSummaryOpinion && stepDefine.StepType is EStepType.Summary)
  2965. opinion = currentStep.Opinion;
  2966. return await CreateStepsAsync(workflow, stepDefine, currentStep, dto,
  2967. assigner, dto.NextHandlers, null,
  2968. EWorkflowStepStatus.WaitForAccept, ECountersignPosition.None,
  2969. true, traceType, null, expiredTime, opinion, newStepConfig, cancellationToken);
  2970. }
  2971. private async Task<List<WorkflowStep>> CreateStepsAsync(
  2972. Workflow workflow,
  2973. StepDefine stepDefine,
  2974. WorkflowStep currentStep,
  2975. BasicWorkflowDto dto,
  2976. //bool isStartCountersign,
  2977. UserInfo assigner,
  2978. List<StepAssignInfo> handlers,
  2979. string? countersignId,
  2980. EWorkflowStepStatus stepStatus,
  2981. ECountersignPosition csPosition,
  2982. bool isOrigin,
  2983. EWorkflowTraceType traceType,
  2984. EHandlerType? handlerType = null,
  2985. DateTime? expiredTime = null,
  2986. string? opinion = null,
  2987. Action<Workflow, WorkflowStep, StepDefine, WorkflowStep>? newStepConfig = null,
  2988. CancellationToken cancellationToken = default
  2989. )
  2990. {
  2991. List<WorkflowStep> steps = new();
  2992. foreach (var handler in handlers)
  2993. {
  2994. var isMain = handlers.Count == 1 || (handlers.Count > 1 && handler.Key == dto.NextMainHandler);
  2995. var step = CreateStep(workflow, stepDefine, currentStep,
  2996. handler, assigner, dto.NextStepCode, countersignId, stepStatus, csPosition, expiredTime,
  2997. dto.NextStepName, isOrigin, isMain, handlerType, dto.BusinessType, dto.FlowDirection, opinion);
  2998. //workflow, currentStep, nextStepDefine, nextStep
  2999. await Task.Run(() => newStepConfig?.Invoke(workflow, currentStep, stepDefine, step), cancellationToken);
  3000. // if (stepConfig != null)
  3001. // {
  3002. // var stepAssignInfo = stepConfig(workflow, stepDefine, prevStep);
  3003. // var validator = new StepAssignInfoValidator();
  3004. // await validator.ValidateAndThrowAsync(stepAssignInfo, cancellationToken);
  3005. // step.Assign(stepAssignInfo);
  3006. // }
  3007. steps.Add(step);
  3008. }
  3009. await _workflowStepRepository.AddRangeAsync(steps, cancellationToken);
  3010. workflow.Steps.AddRange(steps);
  3011. //create traces todo add range traces
  3012. foreach (var step in steps)
  3013. {
  3014. await CreateTraceAsync(workflow, step, traceType, cancellationToken);
  3015. }
  3016. return steps;
  3017. }
  3018. private async Task<WorkflowTrace> CreatePublishTraceAsync(WorkflowTrace endTrace, CancellationToken cancellation)
  3019. {
  3020. var pubTrace = _mapper.Map<WorkflowTrace>(endTrace);
  3021. pubTrace.TraceStyle = ETraceStyle.Publish;
  3022. pubTrace.Name = "中心发布";
  3023. pubTrace.Status = EWorkflowStepStatus.WaitForAccept;
  3024. pubTrace.Code = "publish";
  3025. pubTrace.CreationTime = endTrace.HandleTime ?? DateTime.Now;
  3026. pubTrace.PrevStepId = endTrace.Id;
  3027. pubTrace.PrevStepCode = endTrace.Code;
  3028. pubTrace.PrevStepName = endTrace.Name;
  3029. pubTrace.AcceptorId = null;
  3030. pubTrace.AcceptorName = null;
  3031. pubTrace.AcceptorOrgId = null;
  3032. pubTrace.AcceptorOrgName = null;
  3033. pubTrace.AcceptorOrgAreaCode = null;
  3034. pubTrace.AcceptorOrgAreaName = null;
  3035. pubTrace.AcceptTime = null;
  3036. pubTrace.HandlerId = null;
  3037. pubTrace.HandlerName = null;
  3038. pubTrace.HandlerOrgId = null;
  3039. pubTrace.HandlerOrgName = null;
  3040. pubTrace.HandlerOrgAreaCode = null;
  3041. pubTrace.HandlerOrgAreaName = null;
  3042. pubTrace.HandleTime = null;
  3043. pubTrace.StepExpiredTime = null;
  3044. pubTrace.CreationTime = endTrace.CreationTime.AddSeconds(1);
  3045. await _workflowTraceRepository.AddAsync(pubTrace, cancellation);
  3046. return pubTrace;
  3047. }
  3048. private async Task<WorkflowTrace> CreateVisitTraceAsync(WorkflowTrace pubTrace, UserInfo acceptor, string orderVisitId,
  3049. CancellationToken cancellation)
  3050. {
  3051. if (string.IsNullOrEmpty(orderVisitId))
  3052. throw new UserFriendlyException($"参数异常,orderVisitId不能为空, pubTraceId: {pubTrace.Id}");
  3053. var visitTrace = _mapper.Map<WorkflowTrace>(pubTrace);
  3054. visitTrace.OrderPublishId = null;
  3055. visitTrace.OrderVisitId = orderVisitId;
  3056. visitTrace.TraceStyle = ETraceStyle.Visit;
  3057. visitTrace.Name = "中心回访";
  3058. visitTrace.Status = EWorkflowStepStatus.WaitForAccept;
  3059. visitTrace.Code = "visit";
  3060. visitTrace.CreationTime = pubTrace.HandleTime ?? DateTime.Now;
  3061. visitTrace.PrevStepId = pubTrace.Id;
  3062. visitTrace.PrevStepCode = pubTrace.Code;
  3063. visitTrace.PrevStepName = pubTrace.Name;
  3064. visitTrace.AssignerId = pubTrace.HandlerId;
  3065. visitTrace.AssignerName = pubTrace.HandlerName;
  3066. visitTrace.AssignerOrgId = pubTrace.HandlerOrgId;
  3067. visitTrace.AssignerOrgName = pubTrace.HandlerOrgName;
  3068. visitTrace.AssignerOrgIsCenter = pubTrace.HandlerOrgIsCenter ?? false;
  3069. visitTrace.HandleTime = null;
  3070. visitTrace.AcceptorId = acceptor.UserId;
  3071. visitTrace.AcceptorName = acceptor.UserName;
  3072. visitTrace.AcceptorOrgId = acceptor.OrgId;
  3073. visitTrace.AcceptorOrgName = acceptor.OrgName;
  3074. visitTrace.StepExpiredTime = null;
  3075. await _workflowTraceRepository.AddAsync(visitTrace, cancellation);
  3076. return visitTrace;
  3077. }
  3078. private async Task<WorkflowTrace> CreateTrashEndTraceAsync(WorkflowTrace visitTrace, CancellationToken cancellation)
  3079. {
  3080. var now = DateTime.Now;
  3081. var TrashEndTrace = _mapper.Map<WorkflowTrace>(visitTrace);
  3082. TrashEndTrace.StepId = null;
  3083. TrashEndTrace.OrderPublishId = null;
  3084. TrashEndTrace.OrderVisitId = null;
  3085. TrashEndTrace.TraceStyle = ETraceStyle.TrashEnd;
  3086. TrashEndTrace.Name = "结束";
  3087. TrashEndTrace.Status = EWorkflowStepStatus.Handled;
  3088. TrashEndTrace.Code = "trashend";
  3089. TrashEndTrace.CreationTime = visitTrace.HandleTime ?? now;
  3090. TrashEndTrace.PrevStepId = visitTrace.Id;
  3091. TrashEndTrace.PrevStepCode = visitTrace.Code;
  3092. TrashEndTrace.PrevStepName = visitTrace.Name;
  3093. TrashEndTrace.AssignerId = visitTrace.HandlerId;
  3094. TrashEndTrace.AssignerName = visitTrace.HandlerName;
  3095. TrashEndTrace.AssignerOrgId = visitTrace.HandlerOrgId;
  3096. TrashEndTrace.AssignerOrgName = visitTrace.HandlerOrgName;
  3097. TrashEndTrace.AssignerOrgIsCenter = visitTrace.HandlerOrgIsCenter ?? false;
  3098. TrashEndTrace.AcceptorId = visitTrace.HandlerId;
  3099. TrashEndTrace.AcceptorName = visitTrace.HandlerName;
  3100. TrashEndTrace.AcceptorOrgId = visitTrace.HandlerOrgId;
  3101. TrashEndTrace.AcceptorOrgName = visitTrace.HandlerOrgName;
  3102. TrashEndTrace.AcceptTime = visitTrace.HandleTime ?? now;
  3103. TrashEndTrace.HandlerId = visitTrace.HandlerId;
  3104. TrashEndTrace.HandlerName = visitTrace.HandlerName;
  3105. TrashEndTrace.HandlerOrgId = visitTrace.HandlerOrgId;
  3106. TrashEndTrace.HandlerOrgName = visitTrace.HandlerOrgName;
  3107. TrashEndTrace.HandleTime = visitTrace.HandleTime ?? now;
  3108. TrashEndTrace.StepExpiredTime = null;
  3109. await _workflowTraceRepository.AddAsync(TrashEndTrace, cancellation);
  3110. return TrashEndTrace;
  3111. }
  3112. /// <summary>
  3113. /// 查询未完成节点
  3114. /// </summary>
  3115. public WorkflowStep GetUnHandleStep(List<WorkflowStep> steps, string orgId, string userId, string[] roleIds)
  3116. {
  3117. //var step = GetStep(steps, orgCode, userId, d => d != EWorkflowStepStatus.Handled);
  3118. var step = steps.FirstOrDefault(d => d.IsCanHandle(userId, orgId, roleIds));
  3119. if (step == null)
  3120. throw new UserFriendlyException(
  3121. $"未找到对应节点, workflowId: {steps.FirstOrDefault()?.WorkflowId} orgCode:{orgId}, userId: {userId}, roleIds: {string.Join(',', roleIds)}",
  3122. "未找到对应节点");
  3123. return step;
  3124. }
  3125. /// <summary>
  3126. /// 检查当前办理节点是否为开始节点
  3127. /// </summary>
  3128. /// <param name="workflowId"></param>
  3129. /// <param name="userId">当前办理人Id</param>
  3130. /// <param name="orgId">当前办理人orgId</param>
  3131. /// <param name="cancellationToken"></param>
  3132. /// <returns></returns>
  3133. public async Task<bool> CheckCurrentIsStartStepAsync(string workflowId, string userId, string orgId,
  3134. CancellationToken cancellationToken)
  3135. {
  3136. var workflow = await GetWorkflowAsync(workflowId, withSteps: true, cancellationToken: cancellationToken);
  3137. var currentStep = GetStep(workflow.Steps, orgId, userId, d => d != EWorkflowStepStatus.Handled);
  3138. if (currentStep is null) return false;
  3139. return workflow.Steps.Count == 1 && currentStep.StepType is EStepType.Start && currentStep.IsOrigin;
  3140. }
  3141. /// <summary>
  3142. /// 检查动态节点是否该终止
  3143. /// </summary>
  3144. public bool DynamicShouldTerminal(StepDefine currentStepDefine, int currentOrgLevel)
  3145. {
  3146. if (currentStepDefine.InstanceMode is not EInstanceMode.Dynamic)
  3147. throw new UserFriendlyException("非动态节点");
  3148. switch (currentStepDefine.InstancePolicy)
  3149. {
  3150. case EDynamicPolicy.OrgUpCenterTop:
  3151. case EDynamicPolicy.OrgUp:
  3152. case EDynamicPolicy.ArriveCenter:
  3153. case EDynamicPolicy.ArriveOneOrg:
  3154. case EDynamicPolicy.OrgUpHandleCenterTop:
  3155. case EDynamicPolicy.OrgUpHandle:
  3156. if (!int.TryParse(currentStepDefine.TerminalDynamicMark, out var tMark))
  3157. throw new UserFriendlyException(
  3158. $"TerminalDynamicMark parse to int failed, tMark: {currentStepDefine.TerminalDynamicMark}");
  3159. return currentOrgLevel <= tMark;
  3160. case EDynamicPolicy.OrgUpLeadCenterTop:
  3161. case EDynamicPolicy.OrgUpLead:
  3162. if (!int.TryParse(currentStepDefine.TerminalDynamicMark, out var tMark2))
  3163. throw new UserFriendlyException(
  3164. $"TerminalDynamicMark parse to int failed, tMark: {currentStepDefine.TerminalDynamicMark}");
  3165. var leadRoleCode = _systemSettingCacheManager.GetSetting(SettingConstants.RoleLingDao)?.SettingValue[0];
  3166. var isLead = _sessionContext.Roles.Any(x => x == leadRoleCode);
  3167. return (currentOrgLevel <= tMark2) && (isLead || _sessionContext.OrgIsCenter);
  3168. case EDynamicPolicy.OrgDownCenterTop:
  3169. case EDynamicPolicy.OrgDown:
  3170. if (!int.TryParse(currentStepDefine.TerminalDynamicMark, out var tMark1))
  3171. throw new UserFriendlyException(
  3172. $"TerminalDynamicMark parse to int failed, tMark: {currentStepDefine.TerminalDynamicMark}");
  3173. return currentOrgLevel >= tMark1;
  3174. default:
  3175. throw new ArgumentOutOfRangeException();
  3176. }
  3177. }
  3178. /// <summary>
  3179. /// 终止会签
  3180. /// </summary>
  3181. /// <param name="countersignId"></param>
  3182. /// <param name="cancellationToken"></param>
  3183. /// <returns></returns>
  3184. public async Task<Workflow> TerminalCountersignAsync(string countersignId, DateTime expireTime, CancellationToken cancellationToken)
  3185. {
  3186. var countersign = await _workflowCountersignRepository.GetAsync(countersignId, cancellationToken);
  3187. if (countersign is null)
  3188. throw new UserFriendlyException("无效会签编号");
  3189. //1. 检查会签是否已结束 t: return 2.检查是否有嵌套会签 t: 一起结束 3.结束会签 4.trace 5.检查workflow会签状态,如果会签全结束需更新状态 6.cp会签发起节点变为待办节点
  3190. if (countersign.IsCompleted())
  3191. throw new UserFriendlyException("该会签已结束");
  3192. return await TerminalCountersignAsync(countersign, expireTime, cancellationToken);
  3193. }
  3194. /// <summary>
  3195. /// 终止会签
  3196. /// </summary>
  3197. public async Task<Workflow> TerminalCountersignAsync(WorkflowCountersign countersign, DateTime expireTime,
  3198. CancellationToken cancellationToken)
  3199. {
  3200. // var countersign = await _workflowCountersignRepository.GetAsync(countersignId, cancellationToken);
  3201. // if (countersign is null)
  3202. // throw new UserFriendlyException("无效会签编号");
  3203. //1. 检查会签是否已结束 t: return 2.检查是否有嵌套会签 t: 一起结束 3.结束会签 4.trace 5.检查workflow会签状态,如果会签全结束需更新状态 6.cp会签发起节点变为待办节点
  3204. if (countersign.IsCompleted())
  3205. throw new UserFriendlyException("该会签已结束");
  3206. var workflow = await GetWorkflowAsync(countersign.WorkflowId, withSteps: true, withTraces: true, withDefine: true,
  3207. withCountersigns: true, cancellationToken: cancellationToken);
  3208. if (!workflow.IsInCountersign)
  3209. throw new UserFriendlyException("该流程未处于会签中");
  3210. countersign = workflow.Countersigns.First(d => d.Id == countersign.Id);
  3211. var startCountersignStep = workflow.Steps.Find(d => d.StartCountersignId == countersign.Id);
  3212. if (startCountersignStep is null)
  3213. throw new UserFriendlyException("未查询到发起会签节点");
  3214. if (startCountersignStep.IsStartedCountersignEnd)
  3215. throw new UserFriendlyException("该会签已汇总");
  3216. var updateCountersigns = new List<WorkflowCountersign>();
  3217. EndCountersignWithCascade(countersign, workflow.Countersigns, startCountersignStep.BusinessType, ref updateCountersigns);
  3218. if (updateCountersigns.Any())
  3219. {
  3220. var updateSteps = new List<WorkflowStep>();
  3221. var updateTraces = new List<WorkflowTrace>();
  3222. HandleStepsByTerminalCs(startCountersignStep, workflow.Steps, workflow.Traces, ref updateSteps,
  3223. ref updateTraces);
  3224. if (updateSteps.Any())
  3225. await _workflowStepRepository.RemoveRangeAsync(updateSteps, cancellationToken);
  3226. //await _workflowStepRepository.RemoveNav(updateSteps)
  3227. // .Include(d => d.StepHandlers)
  3228. // .ExecuteCommandAsync();
  3229. if (updateTraces.Any())
  3230. await _workflowTraceRepository.UpdateRangeAsync(updateTraces, cancellationToken);
  3231. await _workflowCountersignRepository.UpdateRangeAsync(updateCountersigns, cancellationToken);
  3232. //cp会签发起节点变为待办节点
  3233. //1. create terminal trace 2. 撤回至startStep
  3234. var stepToDuplicate = startCountersignStep.IsCountersignEndStep
  3235. ? GetCsLoopStartStep(workflow.Steps, startCountersignStep)
  3236. : startCountersignStep;
  3237. // var stepAssignInfo = GetStepAssignInfo(new ReverseFlowStepAssignInfo
  3238. // {
  3239. // ReverseFlowStepCreationPolicy = EReverseFlowStepCreationPolicy.OriginStep
  3240. // }, stepToDuplicate);
  3241. //
  3242. var stepDefine = workflow.WorkflowDefinition.FindStepDefine(stepToDuplicate.Code);
  3243. var newStep = await DuplicateStepWithTraceAsync(workflow, stepDefine, stepToDuplicate, EWorkflowTraceType.Normal, expireTime,
  3244. cancellationToken);
  3245. //当topcsStep结束cs时,实际办理节点应该更新为newStep
  3246. if (startCountersignStep.Id == workflow.TopCountersignStepId)
  3247. {
  3248. workflow.UpdateActualStepWhenAssign(newStep,
  3249. new FlowStepHandler
  3250. {
  3251. UserId = startCountersignStep.HandlerId,
  3252. Username = startCountersignStep.HandlerName,
  3253. OrgId = startCountersignStep.HandlerOrgId,
  3254. OrgName = startCountersignStep.HandlerOrgName,
  3255. RoleId = startCountersignStep.RoleId,
  3256. RoleName = startCountersignStep.RoleName
  3257. });
  3258. workflow.UpdateCurrentStepWhenAssign(newStep,
  3259. new FlowStepHandler
  3260. {
  3261. UserId = startCountersignStep.HandlerId,
  3262. Username = startCountersignStep.HandlerName,
  3263. OrgId = startCountersignStep.HandlerOrgId,
  3264. OrgName = startCountersignStep.HandlerOrgName,
  3265. RoleId = startCountersignStep.RoleId,
  3266. RoleName = startCountersignStep.RoleName
  3267. });
  3268. }
  3269. // //csEndStep又开启了cs,在结束会签时,如果该节点是topcs的end节点, workflow.topcsStep应该更新为前一cs开启stepId
  3270. // if (startCountersignStep.IsTopCountersignEndStep(workflow.TopCountersignStepId))
  3271. // workflow.TopCountersignStepId = startCountersignStep.CountersignStartStepId;
  3272. if (workflow.CheckIfCountersignOver())
  3273. workflow.EndCountersign();
  3274. await _workflowRepository.UpdateAsync(workflow, cancellationToken);
  3275. }
  3276. return workflow;
  3277. }
  3278. private void HandleStepsByTerminalCs(WorkflowStep step, List<WorkflowStep> steps, List<WorkflowTrace> traces,
  3279. ref List<WorkflowStep> updateSteps, ref List<WorkflowTrace> updateTraces)
  3280. {
  3281. if (step.IsStartCountersign)
  3282. {
  3283. var countersignSteps = steps.Where(d => d.CountersignId == step.StartCountersignId).ToList();
  3284. if (countersignSteps.Any())
  3285. {
  3286. foreach (var countersignStep in countersignSteps)
  3287. {
  3288. HandleStepsByTerminalCs(countersignStep, steps, traces, ref updateSteps, ref updateTraces);
  3289. }
  3290. }
  3291. }
  3292. EndStepByTerminalCs(step, traces, ref updateSteps, ref updateTraces);
  3293. }
  3294. private void EndStepByTerminalCs(WorkflowStep step, List<WorkflowTrace> traces,
  3295. ref List<WorkflowStep> updateSteps, ref List<WorkflowTrace> updateTraces)
  3296. {
  3297. var isHandled = step.Status is EWorkflowStepStatus.Handled;
  3298. var opinion = $"会签未办理完成,由 {_sessionContext.OrgName} 的 {_sessionContext.UserName} 终止办理";
  3299. if (step.IsStartCountersign)
  3300. step.CountersignEnd();
  3301. if (step.Status is not EWorkflowStepStatus.Handled)
  3302. {
  3303. step.Handle(_sessionContext.RequiredUserId, _sessionContext.UserName,
  3304. _sessionContext.RequiredOrgId, _sessionContext.OrgName,
  3305. _sessionContext.OrgAreaCode, _sessionContext.OrgAreaName,
  3306. _sessionContext.OrgIsCenter, EHandleMode.Normal, opinion);
  3307. }
  3308. updateSteps.Add(step);
  3309. if (isHandled) return;
  3310. var trace = traces.FirstOrDefault(d => d.StepId == step.Id);
  3311. if (trace != null)
  3312. {
  3313. _mapper.Map(step, trace);
  3314. updateTraces.Add(trace);
  3315. }
  3316. }
  3317. /// <summary>
  3318. /// 结束会签(包含子项)
  3319. /// </summary>
  3320. /// <param name="countersign"></param>
  3321. /// <param name="updateCountersigns"></param>
  3322. private void EndCountersignWithCascade(WorkflowCountersign countersign, List<WorkflowCountersign> countersigns,
  3323. EBusinessType businessType, ref List<WorkflowCountersign> updateCountersigns)
  3324. {
  3325. if (countersign is null) return;
  3326. var childCountersigns = countersigns.Where(d => d.ParentId == countersign.Id).ToList();
  3327. if (childCountersigns.Any())
  3328. {
  3329. foreach (var childCountersign in childCountersigns)
  3330. {
  3331. EndCountersignWithCascade(childCountersign, countersigns, businessType, ref updateCountersigns);
  3332. }
  3333. }
  3334. EndCountersign(countersign, countersigns, businessType, ref updateCountersigns);
  3335. }
  3336. private void EndCountersign(WorkflowCountersign countersign, List<WorkflowCountersign> countersigns,
  3337. EBusinessType businessType, ref List<WorkflowCountersign> updateCountersigns)
  3338. {
  3339. //todo 1. trace? 先确定展现形式 2. end cs
  3340. countersign.End(null, null, businessType,
  3341. _sessionContext.RequiredUserId, _sessionContext.UserName,
  3342. _sessionContext.RequiredOrgId, _sessionContext.OrgName,
  3343. _sessionContext.OrgAreaCode, _sessionContext.OrgAreaName);
  3344. /*
  3345. * //结束step会签信息
  3346. countersignStartStep.CountersignEnd();
  3347. updateSteps.Add(countersignStartStep);
  3348. //结束会签
  3349. currentCountersign.End(currentStep.Id, currentStep.Code, currentStep.BusinessType,
  3350. _sessionContext.RequiredUserId, _sessionContext.UserName,
  3351. _sessionContext.RequiredOrgId, _sessionContext.OrgName,
  3352. _sessionContext.OrgAreaCode, _sessionContext.OrgAreaName);
  3353. await _workflowCountersignRepository.UpdateAsync(currentCountersign, cancellationToken);
  3354. */
  3355. updateCountersigns.Add(countersign);
  3356. }
  3357. private WorkflowStep? GetStep(List<WorkflowStep> steps, string orgCode, string userId,
  3358. Func<EWorkflowStepStatus, bool> predicate) =>
  3359. steps.FirstOrDefault(d =>
  3360. predicate(d.Status) && d.Handlers.Any(x => x.Key == orgCode || x.Key == userId));
  3361. private WorkflowStep CreateStep(
  3362. Workflow workflow,
  3363. StepDefine stepDefine,
  3364. WorkflowStep prevStep,
  3365. // EFlowAssignType? flowAssignType,
  3366. StepAssignInfo handler,
  3367. UserInfo assigner,
  3368. string nextStepCode,
  3369. string? countersignId,
  3370. EWorkflowStepStatus stepStatus,
  3371. ECountersignPosition countersignPosition,
  3372. DateTime? expiredTime,
  3373. string stepName,
  3374. bool isOrigin,
  3375. bool isMainHandler = false,
  3376. EHandlerType? handlerType = null, //动态节点依据动态策略判断
  3377. EBusinessType? businessType = null,
  3378. EFlowDirection? flowDirection = null,
  3379. string? opinion = null
  3380. )
  3381. {
  3382. var step = _mapper.Map<WorkflowStep>(stepDefine);
  3383. _mapper.Map(workflow, step);
  3384. // step.FlowAssignType = flowAssignType;
  3385. step.Handlers = new List<Kv> { new(handler.Key, handler.Value) };
  3386. //step.StepHandlers = stepHandlers;
  3387. step.NextStepCode = step.StepType is EStepType.End ? string.Empty : nextStepCode;
  3388. step.IsMain = isMainHandler;
  3389. step.PrevStepId = prevStep.Id;
  3390. step.PrevStepCode = prevStep.Code;
  3391. step.PrevStepName = prevStep.Name;
  3392. step.CountersignId = countersignId;
  3393. step.Status = stepStatus;
  3394. step.CountersignPosition = countersignPosition;
  3395. step.FlowDirection = flowDirection;
  3396. if (expiredTime.HasValue)
  3397. step.StepExpiredTime = expiredTime;
  3398. //step.TimeLimit = GetTimeLimit("");
  3399. step.IsOrigin = isOrigin;
  3400. if (!string.IsNullOrEmpty(stepName))
  3401. step.Name = stepName;
  3402. //新增需求: 部门汇总节点由部门办理
  3403. //todo 待确认中心由部门处理还是由之前办理人办理 待重构
  3404. if (step.StepType == EStepType.Summary && step.BusinessType == EBusinessType.Department)
  3405. step.FlowAssignType = EFlowAssignType.Org;
  3406. step.Assign(handler);
  3407. if (handlerType.HasValue)
  3408. step.HandlerType = handlerType.Value;
  3409. if (businessType.HasValue)
  3410. step.BusinessType = businessType.Value;
  3411. if (!string.IsNullOrEmpty(opinion))
  3412. step.Opinion = opinion;
  3413. step.AssignerId = assigner.UserId;
  3414. step.AssignerName = assigner.UserName;
  3415. step.AssignerOrgId = assigner.OrgId;
  3416. step.AssignerOrgName = assigner.OrgName;
  3417. step.AssignerOrgIsCenter = assigner.OrgIsCenter;
  3418. return step;
  3419. }
  3420. public async Task<FlowAssignInfo> GetNextStepFlowAssignInfoByDefineAsync(StepDefine nextStepDefine,
  3421. EHandlerType handlerType, bool isStartCountersign, List<Kv> handlers, CancellationToken cancellationToken)
  3422. {
  3423. switch (handlerType)
  3424. {
  3425. case EHandlerType.Role:
  3426. if (!handlers.Any())
  3427. {
  3428. //var roles = await _roleRepository.Queryable()
  3429. // .Includes(d => d.Accounts, x => x.User)
  3430. // .Where(d => nextStepDefine.HandlerTypeItems.Select(x => x.Key).Contains(d.Name))
  3431. // .ToListAsync(cancellationToken);
  3432. //handlers = roles.SelectMany(d => d.Accounts).Distinct()
  3433. // .Select(d => new Kv(d.Id, d.User.Name))
  3434. // .ToList();
  3435. handlers = nextStepDefine.HandlerTypeItems;
  3436. return FlowAssignInfo.Create(EFlowAssignType.Role, handlers, isStartCountersign);
  3437. }
  3438. return FlowAssignInfo.Create(EFlowAssignType.User, handlers, isStartCountersign);
  3439. case EHandlerType.OrgLevel:
  3440. case EHandlerType.OrgType:
  3441. case EHandlerType.AssignedOrg:
  3442. return FlowAssignInfo.Create(EFlowAssignType.Org, handlers, isStartCountersign);
  3443. case EHandlerType.AssignedUser:
  3444. return FlowAssignInfo.Create(EFlowAssignType.User, handlers, isStartCountersign);
  3445. //case EHandlerType.AssignedOrgOrRole:
  3446. // return FlowAssignInfo.Create(EFlowAssignType.OrgAndRole, handlers, isStartCountersign);
  3447. default:
  3448. throw new ArgumentOutOfRangeException();
  3449. }
  3450. }
  3451. //new
  3452. /// <summary>
  3453. /// 查询流程业务模块
  3454. /// </summary>
  3455. /// <param name="code"></param>
  3456. /// <param name="cancellationToken"></param>
  3457. /// <returns></returns>
  3458. /// <exception cref="UserFriendlyException"></exception>
  3459. private async Task<WorkflowModule> GetWorkflowModuleAsync(string code, CancellationToken cancellationToken)
  3460. {
  3461. var wfModule = await _wfModuleCacheManager.GetWorkflowModuleAsync(code, cancellationToken);
  3462. if (wfModule == null)
  3463. throw UserFriendlyException.SameMessage("无效流程模块编码");
  3464. if (wfModule.Definition is null)
  3465. throw new UserFriendlyException($"{code} 未配置流程模板", "未配置流程模板");
  3466. return wfModule;
  3467. }
  3468. /// <summary>
  3469. /// 查询下一节点办理对象类型(user or org)及实际办理对象
  3470. /// </summary>
  3471. private async Task<FlowAssignInfo> GetNextStepFlowAssignInfoAsync(Workflow workflow, WorkflowStep currentStep,
  3472. BasicWorkflowDto dto, StepDefine nextStepDefine, bool isNextDynamic, CancellationToken cancellationToken)
  3473. {
  3474. if (nextStepDefine.StepType is EStepType.End) return new();
  3475. var isStartCountersign = dto.IsStartCountersign;
  3476. var handlers = dto.NextHandlers.Select(d => new Kv(d.Key, d.Value)).ToList();
  3477. if (isStartCountersign)
  3478. {
  3479. var assignType = FlowAssignInfo.GetAssignType(dto.HandlerType, dto.NextHandlers.Any());
  3480. //按会签策略判断,目前所有策略为org
  3481. return FlowAssignInfo.Create(assignType, handlers, isStartCountersign);
  3482. }
  3483. //if (currentStep.IsInCountersign() && !currentStep.IsTopCountersignEndStep(workflow.TopCountersignStepId))
  3484. // return FlowAssignInfo.Create(EFlowAssignType.Org, handlers, isStartCountersign);
  3485. if (currentStep.IsInCountersign())
  3486. {
  3487. if (currentStep.IsCountersignEndStep)
  3488. {
  3489. //汇总节点(非顶级)
  3490. if (!currentStep.IsTopCountersignEndStep(workflow.TopCountersignStepId))
  3491. {
  3492. if (dto.BackToCountersignEnd)
  3493. {
  3494. var csStartStep = GetCsLoopStartStep(workflow.Steps, currentStep);
  3495. var prevStep = workflow.Steps.FirstOrDefault(d => d.Id == csStartStep.PrevStepId);
  3496. if (prevStep is null)
  3497. throw new UserFriendlyException("未查询到目标节点的前一节点");
  3498. return FlowAssignInfo.Create(prevStep.FlowAssignType.Value, prevStep.Handlers, isStartCountersign);
  3499. }
  3500. }
  3501. }
  3502. else
  3503. {
  3504. if (dto.BackToCountersignEnd)
  3505. {
  3506. var prevStep = workflow.Steps.FirstOrDefault(d => d.Id == currentStep.PrevStepId);
  3507. if (prevStep is null)
  3508. throw new UserFriendlyException($"未查询到当前节点的上级节点");
  3509. return FlowAssignInfo.Create(prevStep.FlowAssignType.Value, prevStep.Handlers, isStartCountersign);
  3510. }
  3511. else
  3512. {
  3513. var assignType = FlowAssignInfo.GetAssignType(dto.HandlerType, dto.NextHandlers.Any());
  3514. //按会签策略判断,目前所有策略为org
  3515. return FlowAssignInfo.Create(assignType, handlers, isStartCountersign);
  3516. }
  3517. }
  3518. }
  3519. if (isNextDynamic)
  3520. {
  3521. switch (currentStep.InstancePolicy)
  3522. {
  3523. case EDynamicPolicy.OrgUpCenterTop:
  3524. case EDynamicPolicy.OrgUp:
  3525. case EDynamicPolicy.OrgDownCenterTop:
  3526. case EDynamicPolicy.OrgDown:
  3527. case EDynamicPolicy.ArriveCenter:
  3528. case EDynamicPolicy.ArriveOneOrg:
  3529. return FlowAssignInfo.Create(EFlowAssignType.Org, handlers, isStartCountersign);
  3530. case EDynamicPolicy.OrgUpHandleCenterTop:
  3531. case EDynamicPolicy.OrgUpHandle:
  3532. case EDynamicPolicy.OrgUpLeadCenterTop:
  3533. case EDynamicPolicy.OrgUpLead:
  3534. return FlowAssignInfo.Create(EFlowAssignType.OrgAndRole, handlers, isStartCountersign);
  3535. default:
  3536. throw new ArgumentOutOfRangeException();
  3537. }
  3538. }
  3539. return await GetNextStepFlowAssignInfoByDefineAsync(nextStepDefine, dto.HandlerType, isStartCountersign, handlers,
  3540. cancellationToken);
  3541. }
  3542. private EFlowAssignType SetNextStepAssignInfo(Workflow workflow, WorkflowStep currentStep,
  3543. BasicWorkflowDto dto, StepDefine nextStepDefine, bool isNextDynamic)
  3544. {
  3545. if (nextStepDefine.StepType is EStepType.End) return EFlowAssignType.User;
  3546. if (dto.IsStartCountersign)
  3547. return GetFlowStepAssignType(dto.HandlerType, dto.NextHandlers.Any());
  3548. if (currentStep.IsInCountersign())
  3549. {
  3550. if (currentStep.IsCountersignEndStep)
  3551. {
  3552. //汇总节点(非顶级)
  3553. if (!currentStep.IsTopCountersignEndStep(workflow.TopCountersignStepId))
  3554. {
  3555. if (dto.BackToCountersignEnd)
  3556. {
  3557. var csStartStep = GetCsLoopStartStep(workflow.Steps, currentStep);
  3558. var prevStep = workflow.Steps.FirstOrDefault(d => d.Id == csStartStep.PrevStepId);
  3559. if (prevStep is null)
  3560. throw new UserFriendlyException("未查询到目标节点的前一节点");
  3561. return prevStep.FlowAssignType ?? EFlowAssignType.User;
  3562. }
  3563. }
  3564. }
  3565. else
  3566. {
  3567. if (dto.BackToCountersignEnd)
  3568. {
  3569. var prevStep = workflow.Steps.FirstOrDefault(d => d.Id == currentStep.PrevStepId);
  3570. if (prevStep is null)
  3571. throw new UserFriendlyException($"未查询到当前节点的上级节点");
  3572. return prevStep.FlowAssignType ?? EFlowAssignType.User;
  3573. }
  3574. else
  3575. {
  3576. return GetFlowStepAssignType(dto.HandlerType, dto.NextHandlers.Any());
  3577. }
  3578. }
  3579. }
  3580. if (isNextDynamic)
  3581. {
  3582. switch (currentStep.InstancePolicy)
  3583. {
  3584. case EDynamicPolicy.OrgUpCenterTop:
  3585. case EDynamicPolicy.OrgUp:
  3586. case EDynamicPolicy.OrgDownCenterTop:
  3587. case EDynamicPolicy.OrgDown:
  3588. case EDynamicPolicy.ArriveCenter:
  3589. case EDynamicPolicy.ArriveOneOrg:
  3590. return EFlowAssignType.Org;
  3591. case EDynamicPolicy.OrgUpHandleCenterTop:
  3592. case EDynamicPolicy.OrgUpHandle:
  3593. case EDynamicPolicy.OrgUpLeadCenterTop:
  3594. case EDynamicPolicy.OrgUpLead:
  3595. return EFlowAssignType.OrgAndRole;
  3596. default:
  3597. throw new ArgumentOutOfRangeException();
  3598. }
  3599. }
  3600. return GetFlowStepAssignType(dto.HandlerType, dto.NextHandlers.Any());
  3601. }
  3602. public EFlowAssignType GetFlowStepAssignType(EHandlerType handlerType, bool hasNextHandlers)
  3603. {
  3604. return handlerType switch
  3605. {
  3606. EHandlerType.Role => hasNextHandlers ? EFlowAssignType.User : EFlowAssignType.Role,
  3607. EHandlerType.OrgLevel => EFlowAssignType.Org,
  3608. EHandlerType.OrgType => EFlowAssignType.Org,
  3609. EHandlerType.AssignedUser => EFlowAssignType.User,
  3610. EHandlerType.AssignedOrg => EFlowAssignType.Org,
  3611. _ => throw new ArgumentOutOfRangeException(nameof(handlerType), handlerType, null)
  3612. };
  3613. }
  3614. #endregion
  3615. }
  3616. }