ResourceController.cs 402 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639664066416642664366446645664666476648664966506651665266536654665566566657665866596660666166626663666466656666666766686669667066716672667366746675667666776678667966806681668266836684668566866687668866896690669166926693669466956696669766986699670067016702670367046705670667076708670967106711671267136714671567166717671867196720672167226723672467256726672767286729673067316732673367346735673667376738673967406741674267436744674567466747674867496750675167526753675467556756675767586759676067616762676367646765676667676768676967706771677267736774677567766777677867796780678167826783678467856786678767886789679067916792679367946795679667976798679968006801680268036804680568066807680868096810681168126813681468156816681768186819682068216822682368246825682668276828682968306831683268336834683568366837683868396840684168426843684468456846684768486849685068516852685368546855685668576858685968606861686268636864686568666867686868696870687168726873687468756876687768786879688068816882688368846885688668876888688968906891689268936894689568966897689868996900690169026903690469056906690769086909691069116912691369146915691669176918691969206921692269236924692569266927692869296930693169326933693469356936693769386939694069416942694369446945694669476948694969506951695269536954695569566957695869596960696169626963696469656966696769686969697069716972697369746975697669776978697969806981698269836984698569866987698869896990699169926993699469956996699769986999700070017002700370047005700670077008700970107011701270137014701570167017701870197020702170227023702470257026702770287029703070317032703370347035703670377038703970407041704270437044704570467047704870497050705170527053705470557056705770587059706070617062706370647065706670677068706970707071707270737074707570767077707870797080708170827083708470857086708770887089709070917092709370947095709670977098709971007101710271037104710571067107710871097110711171127113711471157116711771187119712071217122712371247125712671277128712971307131713271337134713571367137713871397140714171427143714471457146714771487149715071517152715371547155715671577158715971607161716271637164716571667167716871697170717171727173717471757176717771787179718071817182718371847185718671877188718971907191719271937194719571967197719871997200720172027203720472057206720772087209721072117212721372147215721672177218721972207221722272237224722572267227722872297230723172327233723472357236723772387239724072417242724372447245724672477248724972507251725272537254725572567257725872597260726172627263726472657266726772687269727072717272727372747275727672777278727972807281728272837284728572867287728872897290729172927293729472957296729772987299730073017302730373047305730673077308730973107311731273137314731573167317731873197320732173227323732473257326732773287329733073317332733373347335733673377338733973407341734273437344734573467347734873497350735173527353735473557356735773587359736073617362736373647365736673677368736973707371737273737374737573767377737873797380738173827383738473857386738773887389739073917392739373947395739673977398739974007401740274037404740574067407740874097410741174127413741474157416741774187419742074217422742374247425742674277428742974307431743274337434743574367437743874397440744174427443744474457446744774487449745074517452745374547455745674577458745974607461746274637464746574667467746874697470747174727473747474757476747774787479748074817482748374847485748674877488748974907491749274937494749574967497749874997500750175027503750475057506750775087509751075117512751375147515751675177518751975207521752275237524752575267527752875297530753175327533753475357536753775387539754075417542754375447545754675477548754975507551755275537554755575567557755875597560756175627563756475657566756775687569757075717572757375747575757675777578757975807581758275837584758575867587758875897590759175927593759475957596759775987599760076017602760376047605760676077608760976107611761276137614761576167617761876197620762176227623762476257626762776287629763076317632763376347635763676377638763976407641764276437644764576467647764876497650765176527653765476557656765776587659766076617662766376647665766676677668766976707671767276737674767576767677767876797680768176827683768476857686768776887689769076917692769376947695769676977698769977007701770277037704770577067707770877097710771177127713771477157716771777187719772077217722772377247725772677277728772977307731773277337734773577367737773877397740774177427743774477457746774777487749775077517752775377547755775677577758775977607761776277637764776577667767776877697770777177727773777477757776777777787779778077817782778377847785778677877788778977907791779277937794779577967797779877997800780178027803780478057806780778087809781078117812781378147815781678177818781978207821782278237824782578267827782878297830783178327833783478357836783778387839784078417842784378447845784678477848784978507851785278537854785578567857785878597860786178627863786478657866786778687869787078717872787378747875787678777878787978807881788278837884788578867887788878897890789178927893789478957896789778987899790079017902790379047905790679077908790979107911791279137914791579167917791879197920792179227923792479257926792779287929793079317932793379347935793679377938793979407941794279437944794579467947794879497950795179527953795479557956795779587959796079617962796379647965796679677968796979707971797279737974797579767977797879797980798179827983798479857986798779887989799079917992799379947995799679977998799980008001800280038004800580068007800880098010801180128013801480158016801780188019802080218022802380248025802680278028802980308031803280338034803580368037803880398040804180428043804480458046804780488049805080518052805380548055805680578058805980608061806280638064806580668067806880698070807180728073807480758076807780788079808080818082808380848085808680878088808980908091809280938094809580968097809880998100810181028103810481058106810781088109811081118112811381148115811681178118811981208121812281238124812581268127812881298130813181328133813481358136813781388139814081418142814381448145814681478148814981508151815281538154815581568157815881598160816181628163816481658166816781688169817081718172817381748175817681778178817981808181818281838184818581868187818881898190819181928193819481958196819781988199820082018202820382048205820682078208820982108211821282138214821582168217821882198220822182228223822482258226822782288229823082318232823382348235823682378238823982408241824282438244824582468247824882498250825182528253825482558256825782588259826082618262826382648265826682678268826982708271827282738274827582768277827882798280828182828283828482858286828782888289829082918292829382948295829682978298829983008301830283038304830583068307830883098310831183128313831483158316831783188319832083218322832383248325832683278328832983308331833283338334833583368337833883398340834183428343834483458346834783488349835083518352835383548355835683578358835983608361836283638364836583668367836883698370837183728373837483758376837783788379838083818382838383848385838683878388838983908391839283938394839583968397839883998400840184028403840484058406840784088409841084118412841384148415841684178418841984208421842284238424842584268427842884298430843184328433843484358436843784388439844084418442844384448445844684478448844984508451845284538454845584568457845884598460846184628463846484658466846784688469847084718472847384748475847684778478847984808481848284838484848584868487848884898490849184928493849484958496849784988499850085018502850385048505850685078508850985108511851285138514851585168517851885198520852185228523852485258526852785288529853085318532853385348535853685378538853985408541854285438544854585468547854885498550855185528553855485558556855785588559856085618562856385648565856685678568856985708571857285738574857585768577857885798580858185828583858485858586858785888589859085918592859385948595859685978598859986008601860286038604860586068607860886098610861186128613861486158616861786188619862086218622862386248625862686278628862986308631863286338634863586368637863886398640864186428643864486458646864786488649865086518652865386548655865686578658865986608661866286638664866586668667866886698670867186728673867486758676867786788679868086818682868386848685868686878688868986908691869286938694869586968697869886998700870187028703870487058706870787088709871087118712871387148715871687178718871987208721872287238724872587268727872887298730873187328733873487358736873787388739874087418742874387448745874687478748874987508751875287538754875587568757875887598760876187628763876487658766876787688769877087718772877387748775877687778778877987808781878287838784878587868787878887898790879187928793879487958796879787988799880088018802880388048805880688078808880988108811881288138814881588168817881888198820882188228823882488258826882788288829883088318832883388348835883688378838883988408841884288438844884588468847884888498850885188528853885488558856885788588859886088618862886388648865886688678868886988708871887288738874887588768877887888798880888188828883888488858886888788888889889088918892889388948895889688978898889989008901890289038904890589068907890889098910891189128913891489158916891789188919892089218922892389248925892689278928892989308931893289338934893589368937893889398940894189428943894489458946894789488949895089518952895389548955895689578958895989608961896289638964896589668967896889698970897189728973897489758976897789788979898089818982898389848985898689878988898989908991899289938994899589968997899889999000900190029003900490059006900790089009901090119012901390149015901690179018901990209021902290239024902590269027902890299030903190329033903490359036903790389039904090419042904390449045904690479048904990509051905290539054905590569057905890599060906190629063906490659066906790689069907090719072907390749075907690779078907990809081908290839084908590869087908890899090909190929093909490959096909790989099910091019102910391049105910691079108910991109111911291139114911591169117911891199120912191229123912491259126912791289129913091319132913391349135913691379138913991409141914291439144914591469147914891499150915191529153915491559156915791589159916091619162916391649165916691679168916991709171917291739174917591769177917891799180918191829183918491859186918791889189919091919192919391949195919691979198919992009201920292039204920592069207920892099210921192129213921492159216921792189219922092219222922392249225922692279228922992309231923292339234923592369237923892399240924192429243924492459246924792489249925092519252925392549255925692579258925992609261926292639264926592669267926892699270927192729273927492759276927792789279928092819282928392849285928692879288928992909291929292939294929592969297929892999300930193029303930493059306930793089309931093119312931393149315931693179318931993209321932293239324932593269327932893299330933193329333933493359336933793389339934093419342934393449345934693479348934993509351935293539354935593569357935893599360936193629363936493659366936793689369937093719372937393749375937693779378937993809381938293839384938593869387938893899390939193929393939493959396939793989399940094019402940394049405940694079408940994109411941294139414941594169417941894199420942194229423942494259426942794289429943094319432943394349435943694379438943994409441944294439444944594469447944894499450945194529453945494559456945794589459946094619462946394649465946694679468946994709471947294739474947594769477947894799480948194829483948494859486948794889489949094919492949394949495949694979498949995009501950295039504950595069507950895099510951195129513951495159516951795189519
  1. using Aspose.Cells;
  2. using Aspose.Cells.Drawing;
  3. using Aspose.Words;
  4. using Aspose.Words.Tables;
  5. using Azure;
  6. using EyeSoft.Extensions;
  7. using Microsoft.AspNetCore.Http.Features;
  8. using Microsoft.Graph.Models;
  9. using Microsoft.Kiota.Abstractions;
  10. using Newtonsoft.Json.Serialization;
  11. using NodaTime;
  12. using NPOI.SS.Formula.Functions;
  13. using NPOI.SS.UserModel;
  14. using OASystem.API.OAMethodLib;
  15. using OASystem.API.OAMethodLib.Hotmail;
  16. using OASystem.API.OAMethodLib.HunYuanAPI;
  17. using OASystem.API.OAMethodLib.QiYeWeChatAPI;
  18. using OASystem.API.OAMethodLib.QiYeWeChatAPI.AppNotice;
  19. using OASystem.Domain.AesEncryption;
  20. using OASystem.Domain.Dtos.Groups;
  21. using OASystem.Domain.Entities.Groups;
  22. using OASystem.Domain.ViewModels.Groups;
  23. using OASystem.Domain.ViewModels.QiYeWeChat;
  24. using OASystem.Domain.ViewModels.Statistics;
  25. using OASystem.Infrastructure.Repositories.Groups;
  26. using Quartz.Logging;
  27. using Quartz.Util;
  28. using System.ComponentModel.DataAnnotations;
  29. using System.Data;
  30. using System.Diagnostics;
  31. using System.Security.AccessControl;
  32. using TencentCloud.Common;
  33. using static NodaTime.TimeZones.TzdbZone1970Location;
  34. using static OASystem.API.Controllers.PersonnelModuleController;
  35. using static OASystem.API.OAMethodLib.GeneralMethod;
  36. using static OASystem.API.OAMethodLib.Hotmail.HotmailService;
  37. using static OASystem.API.OAMethodLib.JWTHelper;
  38. using static OpenAI.GPT3.ObjectModels.Models;
  39. using Workbook = Aspose.Cells.Workbook;
  40. namespace OASystem.API.Controllers
  41. {
  42. /// <summary>
  43. /// 资料相关
  44. /// </summary>
  45. //[Authorize]
  46. [Route("api/[controller]/[action]")]
  47. public class ResourceController : ControllerBase
  48. {
  49. private readonly IMapper _mapper;
  50. private readonly ILogger<ResourceController> _logger;
  51. private readonly IConfiguration _config;
  52. private readonly IHunyuanService _hunyuanService;
  53. private readonly IQiYeWeChatApiService _qiYeWeChatApiService;
  54. private readonly SqlSugarClient _sqlSugar;
  55. private readonly CarDataRepository _carDataRep;
  56. private readonly LocalGuideDataRepository _localGuideDataRep;
  57. private readonly ThreeCodeRepository _ThreeCodeRep;
  58. private readonly HotelDataRepository _hotelDataRep;
  59. private readonly ResItemInfoRepository _resItemInfoRep;
  60. private readonly SetDataRepository _setDataRepository;
  61. private readonly CountryFeeRepository _countryFeeRep;
  62. private readonly SetDataTypeRepository _setDataTypeRep;
  63. private readonly AirTicketAgentRepository _airTicketAgentRep;
  64. private readonly InvitationOfficialActivityDataRepository _InvitationOfficialActivityDataRep;
  65. private readonly OfficialActivitiesRepository _officialActivitiesRep;
  66. private readonly AskDataRepository _askDataRep;
  67. private readonly TicketBlackCodeRepository _ticketBlackCodeRep;
  68. private readonly TourClientListRepository _tourClientListRep;
  69. private readonly DelegationInfoRepository _delegationInfoRep;
  70. private readonly TranslatorLibraryRepository _translatorRep;
  71. private readonly MediaSuppliersRepository _mediaSupplierRep;
  72. private readonly List<int> _portIds;
  73. private readonly BasicInsuranceCostRepository _insuranceCostRep;
  74. private readonly GamesBudgetMasterRepository _gamesBudgetMasterRep;
  75. private readonly OverseaVehicleRepository _overseaVehicleRep;
  76. private readonly MaterialCostRepository _materialCostRep;
  77. private readonly HotmailService _hotmailService;
  78. /// <summary>
  79. /// 签证费用归属省份静态数据
  80. /// </summary>
  81. private static readonly List<KeyValuePair<int, string>> _provinceData = new()
  82. {
  83. new(122, "四川"),
  84. new(132, "云南"),
  85. new(103, "重庆"),
  86. new(108, "贵州")
  87. };
  88. public ResourceController(
  89. IMapper mapper,
  90. IConfiguration config,
  91. ILogger<ResourceController> logger,
  92. IHunyuanService hunyuanService,
  93. IQiYeWeChatApiService qiYeWeChatApiService,
  94. HotmailService hotmailService,
  95. SqlSugarClient sqlSugar,
  96. CarDataRepository carDataRep,
  97. LocalGuideDataRepository localGuideDataRep,
  98. ThreeCodeRepository threeCodeRep,
  99. HotelDataRepository hotelDataRep,
  100. ResItemInfoRepository resItemInfoRep,
  101. SetDataRepository setDataRepository,
  102. CountryFeeRepository countryFeeRep,
  103. SetDataTypeRepository setDataTypeRep,
  104. AirTicketAgentRepository airTicketAgentRep,
  105. InvitationOfficialActivityDataRepository invitationOfficialActivityDataRep,
  106. OfficialActivitiesRepository officialActivitiesRep,
  107. AskDataRepository askDataRep,
  108. TicketBlackCodeRepository ticketBlackCodeRep,
  109. TourClientListRepository tourClientListRep,
  110. DelegationInfoRepository delegationInfoRep,
  111. TranslatorLibraryRepository translatorRep,
  112. MediaSuppliersRepository mediaSupplierRep,
  113. BasicInsuranceCostRepository insuranceCostRep,
  114. GamesBudgetMasterRepository gamesBudgetMasterRep,
  115. OverseaVehicleRepository overseaVehicleRep,
  116. MaterialCostRepository materialCostRep
  117. )
  118. {
  119. _mapper = mapper;
  120. _config = config;
  121. _logger = logger;
  122. _hunyuanService = hunyuanService;
  123. _qiYeWeChatApiService = qiYeWeChatApiService;
  124. _hotmailService = hotmailService;
  125. _sqlSugar = sqlSugar;
  126. _carDataRep = carDataRep;
  127. _localGuideDataRep = localGuideDataRep;
  128. _ThreeCodeRep = threeCodeRep;
  129. _hotelDataRep = hotelDataRep;
  130. _resItemInfoRep = resItemInfoRep;
  131. _setDataRepository = setDataRepository;
  132. _countryFeeRep = countryFeeRep;
  133. _setDataTypeRep = setDataTypeRep;
  134. _airTicketAgentRep = airTicketAgentRep;
  135. _InvitationOfficialActivityDataRep = invitationOfficialActivityDataRep;
  136. _officialActivitiesRep = officialActivitiesRep;
  137. _askDataRep = askDataRep;
  138. _ticketBlackCodeRep = ticketBlackCodeRep;
  139. _tourClientListRep = tourClientListRep;
  140. _delegationInfoRep = delegationInfoRep;
  141. _translatorRep = translatorRep;
  142. _mediaSupplierRep = mediaSupplierRep;
  143. _portIds = new List<int> { 1, 2, 3 };
  144. _insuranceCostRep = insuranceCostRep;
  145. _gamesBudgetMasterRep = gamesBudgetMasterRep;
  146. _overseaVehicleRep = overseaVehicleRep;
  147. _materialCostRep = materialCostRep;
  148. }
  149. #region 车公司资料板块
  150. /// <summary>
  151. /// 车公司信息查询
  152. /// </summary>
  153. /// <returns></returns>
  154. [HttpPost]
  155. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  156. public async Task<IActionResult> QueryCarData(QueryCarDataDto dto)
  157. {
  158. Result LocalGuide = await _carDataRep.QueryCarData(dto);
  159. if (LocalGuide.Code == 0)
  160. {
  161. return Ok(JsonView(true, "查询成功", LocalGuide.Data));
  162. }
  163. else
  164. {
  165. return Ok(JsonView(false, LocalGuide.Msg));
  166. }
  167. }
  168. /// <summary>
  169. /// 车公司资料下拉框数据
  170. /// </summary>
  171. /// <returns></returns>
  172. [HttpPost]
  173. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  174. public async Task<IActionResult> QueryCarSelect()
  175. {
  176. var carData = _carDataRep.QueryDto<Res_CarData, CarDataSelectView>().ToList();
  177. if (carData.Count == 0)
  178. {
  179. return Ok(JsonView(false, "暂无数据!"));
  180. }
  181. foreach (var item in carData) EncryptionProcessor.DecryptProperties(item);
  182. carData.Add(new CarDataSelectView { Id = 0, UnitArea = "全部" });
  183. carData = carData.Where((x, i) => carData.FindIndex(z => z.UnitArea == x.UnitArea) == i).ToList();
  184. carData = carData.OrderBy(x => x.Id).ToList();
  185. var data = new List<CarDataSelectView>();
  186. foreach (CarDataSelectView car in carData)
  187. {
  188. if (!string.IsNullOrWhiteSpace(car.UnitArea))
  189. {
  190. data.Add(car);
  191. }
  192. }
  193. return Ok(JsonView(true, "查询成功", data));
  194. }
  195. /// <summary>
  196. /// 根据Id查询车公司详细数据
  197. /// </summary>
  198. /// <returns></returns>
  199. [HttpPost]
  200. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  201. public async Task<IActionResult> QuerCarDataDetailById(QueryCarDataDetailDto dto)
  202. {
  203. string sql = string.Format(@" Select * From Res_CarData With(Nolock) Where Id = {0} ", dto.Id);
  204. var info = _resItemInfoRep._sqlSugar.SqlQueryable<CarDataDetailDataView>(sql).First();
  205. if (info == null)
  206. {
  207. return Ok(JsonView(false, "未找到相关数据!"));
  208. }
  209. EncryptionProcessor.DecryptProperties(info);
  210. return Ok(JsonView(true, "查询成功", info));
  211. }
  212. /// <summary>
  213. /// 车公司信息添加
  214. /// </summary>
  215. /// <returns></returns>
  216. [HttpPost]
  217. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  218. public async Task<IActionResult> AddCarData(AddCarDataDto dto)
  219. {
  220. if (string.IsNullOrEmpty(dto.UnitArea)) return Ok(JsonView(false, "请检查单位区域是否填写!"));
  221. if (string.IsNullOrEmpty(dto.UnitName)) return Ok(JsonView(false, "请检查单位名称是否填写!"));
  222. if (string.IsNullOrEmpty(dto.Contact)) return Ok(JsonView(false, "请检查单位联系人是否填写!"));
  223. if (string.IsNullOrEmpty(dto.ContactTel)) return Ok(JsonView(false, "请检查联系方式是否填写正确!"));
  224. var carDadaInfo = _carDataRep
  225. .QueryDto<Res_CarData, CarDataView>(a => a.UnitArea == dto.UnitArea &&
  226. a.UnitName == dto.UnitName &&
  227. a.Contact == dto.Contact &&
  228. a.ContactTel == dto.ContactTel
  229. )
  230. .First();
  231. if (carDadaInfo != null) return Ok(JsonView(false, "该信息已存在,请勿重复添加!"));
  232. var carData = _mapper.Map<Res_CarData>(dto);
  233. carData.LastUpdateUserId = dto.CreateUserId;
  234. carData.LastUpdateTime = DateTime.Now;
  235. EncryptionProcessor.EncryptProperties(carData);
  236. int id = await _carDataRep.AddAsyncReturnId(carData);
  237. if (id < 1) return Ok(JsonView(false, "添加失败!"));
  238. return Ok(JsonView(true, "添加成功", new { Id = id }));
  239. }
  240. /// <summary>
  241. /// 车公司信息修改
  242. /// </summary>
  243. /// <returns></returns>
  244. [HttpPost]
  245. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  246. public async Task<IActionResult> UpCarData(UpCarDataDto dto)
  247. {
  248. if (string.IsNullOrEmpty(dto.UnitArea)) return Ok(JsonView(false, "请检查单位区域是否填写!"));
  249. if (string.IsNullOrEmpty(dto.UnitName)) return Ok(JsonView(false, "请检查单位名称是否填写!"));
  250. if (string.IsNullOrEmpty(dto.Contact)) return Ok(JsonView(false, "请检查单位联系人是否填写!"));
  251. if (string.IsNullOrEmpty(dto.ContactTel)) return Ok(JsonView(false, "请检查联系方式是否填写正确!"));
  252. var carData = _mapper.Map<Res_CarData>(dto);
  253. carData.LastUpdateTime = DateTime.Now;
  254. EncryptionProcessor.EncryptProperties(carData);
  255. var res = await _sqlSugar.Updateable(carData)
  256. .IgnoreColumns(x => new { x.CreateUserId, x.CreateTime, x.DeleteTime, x.DeleteUserId, x.IsDel })
  257. .ExecuteCommandAsync();
  258. if (res < 1) { return Ok(JsonView(false, "修改失败!")); }
  259. return Ok(JsonView(true, "修改成功"));
  260. }
  261. /// <summary>
  262. /// 车公司信息删除
  263. /// </summary>
  264. /// <returns></returns>
  265. [HttpPost]
  266. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  267. public async Task<IActionResult> DelCarData(DelCarDataDto dto)
  268. {
  269. bool res = await _carDataRep.SoftDeleteByIdAsync<Res_CarData>(dto.Id.ToString(), dto.DeleteUserId);
  270. if (!res) { return Ok(JsonView(false, "删除失败!")); }
  271. return Ok(JsonView(true, "删除成功"));
  272. }
  273. #endregion
  274. #region 导游地接资料板块
  275. [HttpPost]
  276. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  277. public async Task<IActionResult> LocalGuideBatchEncryption()
  278. {
  279. var infos = await _sqlSugar.Queryable<Res_LocalGuideData>().ToListAsync();
  280. foreach (var item in infos) EncryptionProcessor.EncryptProperties(item);
  281. await _sqlSugar.Updateable(infos).ExecuteCommandAsync();
  282. return Ok(JsonView(true));
  283. }
  284. /// <summary>
  285. /// 导游地接资料查询
  286. /// </summary>
  287. /// <param name="dto"></param>
  288. /// <returns></returns>
  289. [HttpPost]
  290. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  291. public async Task<IActionResult> QueryLocalGuide(QueryLocalGuide dto)
  292. {
  293. Stopwatch stopwatch = Stopwatch.StartNew();
  294. Result LocalGuide = await _localGuideDataRep.QueryLocalGuide(dto);
  295. if (LocalGuide.Code == 0)
  296. {
  297. stopwatch.Stop();
  298. return Ok(JsonView(true, $"查询成功!耗时:{stopwatch.ElapsedMilliseconds} 毫秒", LocalGuide.Data));
  299. }
  300. stopwatch.Stop();
  301. return Ok(JsonView(false, LocalGuide.Msg + $" 耗时:{stopwatch.ElapsedMilliseconds} 毫秒"));
  302. }
  303. /// <summary>
  304. /// 导游地接资料下拉框数据
  305. /// </summary>
  306. /// <returns></returns>
  307. [HttpPost]
  308. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  309. public async Task<IActionResult> QueryLocalGuideSelect()
  310. {
  311. var localGuides = _carDataRep.QueryDto<Res_LocalGuideData, QueryLocalGuideSelectView>().ToList();
  312. if (localGuides.Count == 0)
  313. {
  314. return Ok(JsonView(false, "暂无数据!"));
  315. }
  316. foreach (var item in localGuides) EncryptionProcessor.DecryptProperties(item);
  317. localGuides.Add(new QueryLocalGuideSelectView { Id = 0, UnitArea = "全部" });
  318. localGuides = localGuides.Where((x, i) => localGuides.FindIndex(z => z.UnitArea == x.UnitArea && !string.IsNullOrEmpty(z.UnitArea)) == i).ToList();
  319. localGuides = localGuides.OrderBy(x => x.Id).ToList();
  320. var data = new List<QueryLocalGuideSelectView>();
  321. foreach (QueryLocalGuideSelectView Local in localGuides)
  322. {
  323. if (!string.IsNullOrWhiteSpace(Local.UnitArea))
  324. {
  325. data.Add(Local);
  326. }
  327. }
  328. return Ok(JsonView(true, "查询成功", data));
  329. }
  330. /// <summary>
  331. /// 根据Id查询地接详细数据
  332. /// </summary>
  333. /// <returns></returns>
  334. [HttpPost]
  335. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  336. public async Task<IActionResult> QueryLocalGuideDetailById(QueryLocalGuideDetailDto dto)
  337. {
  338. string sql = string.Format(@" Select * From Res_LocalGuideData With(Nolock) Where Id = {0} ", dto.Id);
  339. var info = _resItemInfoRep._sqlSugar.SqlQueryable<LocalGuideDetailDataView>(sql).First();
  340. if (info == null)
  341. {
  342. return Ok(JsonView(false, "未找到相关数据!"));
  343. }
  344. EncryptionProcessor.DecryptProperties(info);
  345. return Ok(JsonView(true, "查询成功", info));
  346. }
  347. /// <summary>
  348. /// 导游地接信息操作(Status:1.新增,2.修改)
  349. /// </summary>
  350. /// <returns></returns>
  351. [HttpPost]
  352. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  353. public async Task<IActionResult> OperationLocalGuide(LocalGuideOperationDto dto)
  354. {
  355. if (string.IsNullOrEmpty(dto.UnitArea)) return Ok(JsonView(false, "请检查单位区域是否填写!"));
  356. if (string.IsNullOrEmpty(dto.UnitName)) return Ok(JsonView(false, "请检查单位名称是否填写!"));
  357. if (string.IsNullOrEmpty(dto.Contact)) return Ok(JsonView(false, "请检查单位联系人是否填写!"));
  358. if (string.IsNullOrEmpty(dto.ContactTel)) return Ok(JsonView(false, "请检查联系方式是否填写正确!"));
  359. Result result = await _localGuideDataRep.LocalGuideOperation(dto);
  360. if (result.Code != 0)
  361. {
  362. return Ok(JsonView(false, result.Msg));
  363. }
  364. return Ok(JsonView(true, result.Msg));
  365. }
  366. /// <summary>
  367. /// 导游地接信息操作(删除)
  368. /// </summary>
  369. /// <returns></returns>
  370. [HttpPost]
  371. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  372. public async Task<IActionResult> DelLocalGuide(LocalGuideDelDto dto)
  373. {
  374. var res = await _localGuideDataRep.SoftDeleteByIdAsync<Res_LocalGuideData>(dto.Id.ToString(), dto.DeleteUserId);
  375. if (!res)
  376. {
  377. return Ok(JsonView(false, "删除失败"));
  378. }
  379. return Ok(JsonView(true, "删除成功!"));
  380. }
  381. #endregion
  382. #region 机场三字码信息
  383. /// <summary>
  384. /// 机场三字码查询
  385. /// </summary>
  386. /// <param name="dto"></param>
  387. /// <returns></returns>
  388. [HttpPost]
  389. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  390. public async Task<IActionResult> QueryThreeCode(QueryThreeCodeDto dto)
  391. {
  392. try
  393. {
  394. Result LocalGuide = await _ThreeCodeRep.QueryThreeCode(dto);
  395. if (LocalGuide.Code == 0)
  396. {
  397. return Ok(JsonView(true, "查询成功", LocalGuide.Data));
  398. }
  399. else
  400. {
  401. return Ok(JsonView(false, LocalGuide.Msg));
  402. }
  403. }
  404. catch (Exception ex)
  405. {
  406. return Ok(JsonView(false, "程序错误!"));
  407. throw;
  408. }
  409. }
  410. /// <summary>
  411. /// 机场三字码数据城市下拉框数据
  412. /// </summary>
  413. /// <returns></returns>
  414. [HttpPost]
  415. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  416. public async Task<IActionResult> QueryThreeCodeSelect()
  417. {
  418. try
  419. {
  420. var ThreeCode = _carDataRep.QueryDto<Res_ThreeCode, ThreeCodeSelectView>().ToList();
  421. if (ThreeCode.Count == 0)
  422. {
  423. return Ok(JsonView(false, "暂无数据!"));
  424. }
  425. ThreeCode.Add(new ThreeCodeSelectView { Id = 0, City = "全部" });
  426. ThreeCode = ThreeCode.Where((x, i) => ThreeCode.FindIndex(z => z.City == x.City && z.City != "") == i).ToList();
  427. ThreeCode = ThreeCode.OrderBy(x => x.Id).ToList();
  428. List<ThreeCodeSelectView> data = new List<ThreeCodeSelectView>();
  429. foreach (ThreeCodeSelectView _ThreeCode in ThreeCode)
  430. {
  431. if (!string.IsNullOrWhiteSpace(_ThreeCode.City))
  432. {
  433. data.Add(_ThreeCode);
  434. }
  435. }
  436. return Ok(JsonView(true, "查询成功", data));
  437. }
  438. catch (Exception)
  439. {
  440. return Ok(JsonView(false, "程序错误!"));
  441. throw;
  442. }
  443. }
  444. /// <summary>
  445. /// 机场三字码资料操作(Status:1.新增,2.修改)
  446. /// </summary>
  447. /// <param name="dto"></param>
  448. /// <returns></returns>
  449. [HttpPost]
  450. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  451. public async Task<IActionResult> OperationThreeCode(ThreeCodeOperationDto dto)
  452. {
  453. try
  454. {
  455. if (dto.Three == "")
  456. {
  457. return Ok(JsonView(false, "请检查三字码是否填写!"));
  458. }
  459. if (dto.Country == "")
  460. {
  461. return Ok(JsonView(false, "请检查国家是否填写!"));
  462. }
  463. if (dto.City == "")
  464. {
  465. return Ok(JsonView(false, "请检查城市是否填写正确!"));
  466. }
  467. if (dto.AirPort == "")
  468. {
  469. return Ok(JsonView(false, "请检查机场是否填写正确!"));
  470. }
  471. Result result = await _ThreeCodeRep.ThreeCodeOperation(dto);
  472. if (result.Code != 0)
  473. {
  474. return Ok(JsonView(false, result.Msg));
  475. }
  476. return Ok(JsonView(true, result.Msg));
  477. }
  478. catch (Exception ex)
  479. {
  480. return Ok(JsonView(false, "程序错误!"));
  481. throw;
  482. }
  483. }
  484. /// <summary>
  485. /// 机场三字码资料操作(删除)
  486. /// </summary>
  487. /// <returns></returns>
  488. [HttpPost]
  489. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  490. public async Task<IActionResult> DelThreeCode(ThreeCodeDelDto dto)
  491. {
  492. try
  493. {
  494. var res = await _ThreeCodeRep.SoftDeleteByIdAsync<Res_ThreeCode>(dto.Id.ToString(), dto.DeleteUserId);
  495. if (!res)
  496. {
  497. return Ok(JsonView(false, "删除失败"));
  498. }
  499. return Ok(JsonView(true, "删除成功!"));
  500. }
  501. catch (Exception ex)
  502. {
  503. return Ok(JsonView(false, "程序错误!"));
  504. throw;
  505. }
  506. }
  507. /// <summary>
  508. ///根据ID查询三字码信息
  509. /// </summary>
  510. /// <returns></returns>
  511. [HttpPost]
  512. public IActionResult QuerySingleThreeCode(QuerySingleThreeCode dto)
  513. {
  514. var jw = JsonView(false);
  515. var single = _sqlSugar.Queryable<Res_ThreeCode>().First(x => x.Id == dto.ID && x.IsDel == 0);
  516. if (single != null)
  517. {
  518. jw = JsonView(true, "获取成功!", new
  519. {
  520. single.Three,
  521. Four = single.Four?.Trim(),
  522. single.AirPort,
  523. single.AirPort_En,
  524. single.City,
  525. single.Country,
  526. single.Remark
  527. });
  528. }
  529. else
  530. {
  531. jw.Msg = "暂无!";
  532. }
  533. return Ok(jw);
  534. }
  535. #endregion
  536. #region 代理出票合作方资料
  537. /// <summary>
  538. /// 代理出票合作方资料
  539. /// </summary>
  540. /// <param name="dto"></param>
  541. /// <returns></returns>
  542. [HttpPost]
  543. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  544. public async Task<IActionResult> QueryAirTicketAgent(DtoBase dto)
  545. {
  546. var res_AirTicketAgents = _airTicketAgentRep.Query<Res_AirTicketAgent>(a => a.IsDel == 0).ToList();
  547. if (res_AirTicketAgents.Count > 1)
  548. {
  549. res_AirTicketAgents = res_AirTicketAgents.OrderByDescending(a => a.CreateTime).ToList();
  550. if (dto.PageSize == 0 && dto.PageIndex == 0)
  551. {
  552. foreach (var item in res_AirTicketAgents) EncryptionProcessor.DecryptProperties(item);
  553. return Ok(JsonView(true, "查询成功!", res_AirTicketAgents));
  554. }
  555. else
  556. {
  557. int count = res_AirTicketAgents.Count;
  558. float totalPage = (float)count / dto.PageSize;//总页数
  559. if (totalPage == 0) totalPage = 1;
  560. else totalPage = (int)Math.Ceiling((double)totalPage);
  561. var _AirTicketAgent = new List<Res_AirTicketAgent>();
  562. for (int i = 0; i < dto.PageSize; i++)
  563. {
  564. var RowIndex = i + (dto.PageIndex - 1) * dto.PageSize;
  565. if (RowIndex < res_AirTicketAgents.Count)
  566. {
  567. EncryptionProcessor.DecryptProperties(res_AirTicketAgents[RowIndex]);
  568. _AirTicketAgent.Add(res_AirTicketAgents[RowIndex]);
  569. }
  570. else break;
  571. }
  572. return Ok(JsonView(true, "查询成功!", new { pageCount = count, totalPage = (int)totalPage, pageIndex = dto.PageIndex, pageSize = dto.PageSize, pageSource = _AirTicketAgent }));
  573. }
  574. }
  575. return Ok(JsonView(false, "暂无数据!"));
  576. }
  577. /// <summary>
  578. /// 代理出票合作方资料操作(Status:1.新增,2.修改)
  579. /// </summary>
  580. /// <param name="dto"></param>
  581. /// <returns></returns>
  582. [HttpPost]
  583. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  584. public async Task<IActionResult> OpAirTicketAgent(OpAirTicketAgentDto dto)
  585. {
  586. if (string.IsNullOrEmpty(dto.Account)) return Ok(JsonView(false, "请检查代理商账户是否填写!"));
  587. if (string.IsNullOrEmpty(dto.Bank)) return Ok(JsonView(false, "请检查代理商银行是否填写!"));
  588. if (string.IsNullOrEmpty(dto.Name)) return Ok(JsonView(false, "请检查代理商名称是否填写正确!"));
  589. Result result = await _airTicketAgentRep.OpAirTicketAgent(dto);
  590. if (result.Code != 0) return Ok(JsonView(false, result.Msg));
  591. return Ok(JsonView(true, result.Msg));
  592. }
  593. /// <summary>
  594. /// 代理出票合作方资料操作(删除)
  595. /// </summary>
  596. /// <returns></returns>
  597. [HttpPost]
  598. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  599. public async Task<IActionResult> DelAirTicketAgent(DelBaseDto dto)
  600. {
  601. var res = await _airTicketAgentRep.SoftDeleteByIdAsync<Res_AirTicketAgent>(dto.Id.ToString(), dto.DeleteUserId);
  602. if (!res) return Ok(JsonView(false, "删除失败"));
  603. return Ok(JsonView(true, "删除成功!"));
  604. }
  605. #endregion
  606. #region 酒店资料数据
  607. /// <summary>
  608. /// 酒店信息查询 Page
  609. /// </summary>
  610. /// <param name="dto"></param>
  611. /// <returns></returns>
  612. [HttpPost]
  613. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  614. public async Task<IActionResult> QueryHotelData(QueryHotelDataDto dto)
  615. {
  616. string sqlWhere = string.Empty;
  617. if (!string.IsNullOrWhiteSpace(dto.Name))
  618. {
  619. sqlWhere += string.Format(@" And hd.Name like '%{0}%'", AesEncryptionHelper.Encrypt(dto.Name));
  620. }
  621. if (!string.IsNullOrWhiteSpace(dto.City) && dto.City != "全部")
  622. {
  623. sqlWhere += string.Format(@" And hd.City like '%{0}%'", AesEncryptionHelper.Encrypt(dto.City));
  624. }
  625. if (!string.IsNullOrWhiteSpace(dto.Contact))
  626. {
  627. sqlWhere += string.Format(@" And hd.Contact like '%{0}%'", AesEncryptionHelper.Encrypt(dto.Contact));
  628. }
  629. if (!string.IsNullOrWhiteSpace(dto.ContactPhone))
  630. {
  631. sqlWhere += string.Format(@" And hd.ContactPhone like '%{0}%'", AesEncryptionHelper.Encrypt(dto.ContactPhone));
  632. }
  633. sqlWhere += string.Format(@" And hd.IsDel={0}", 0);
  634. if (!string.IsNullOrEmpty(sqlWhere.Trim()))
  635. {
  636. Regex r = new Regex("And");
  637. sqlWhere = r.Replace(sqlWhere, "Where", 1);
  638. }
  639. string sql = string.Format(@"SELECT
  640. *
  641. FROM
  642. (
  643. SELECT
  644. ROW_NUMBER() OVER (
  645. ORDER BY
  646. hd.CreateTime DESC
  647. ) AS Row_Number,
  648. hd.Id,
  649. hd.City,
  650. hd.[Name],
  651. hd.Tel,
  652. hd.Fax,
  653. hd.Contact,
  654. hd.ContactPhone,
  655. u.CnName AS CreateUserName,
  656. hd.CreateTime
  657. FROM
  658. Res_HotelData hd
  659. WITH
  660. (NoLock)
  661. LEFT JOIN Sys_Users u
  662. WITH
  663. (NoLock) ON hd.CreateUserId = u.Id {0}
  664. ) Temp
  665. ", sqlWhere);
  666. if (dto.PortType == 1)
  667. {
  668. var HotelDataData = await _sqlSugar.SqlQueryable<HotelDataItemView>(sql).ToListAsync();
  669. if (HotelDataData.Count == 0)
  670. {
  671. return Ok(JsonView(false, "暂无数据"));
  672. }
  673. foreach (var item in HotelDataData)
  674. {
  675. EncryptionProcessor.DecryptProperties(item);
  676. }
  677. return Ok(JsonView(true, "操作成功", HotelDataData));
  678. }
  679. else if (dto.PortType == 2 || dto.PortType == 3)
  680. {
  681. RefAsync<int> total = 0;//REF和OUT不支持异步,想要真的异步这是最优解
  682. var data = await _sqlSugar.SqlQueryable<HotelDataItemView>(sql).ToPageListAsync(dto.PageIndex, dto.PageSize, total); //ToPageAsync
  683. foreach (var item in data) EncryptionProcessor.DecryptProperties(item);
  684. return Ok(JsonView(true, "操作成功", data, total));
  685. }
  686. else return Ok(JsonView(false, "请传入PortType参数!1:Web,2:Android,3:IOS"));
  687. }
  688. /// <summary>
  689. /// 酒店资料
  690. /// 详情
  691. /// add time:2024-05-14 11:57:10
  692. /// </summary>
  693. /// <returns></returns>
  694. [HttpPost]
  695. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  696. public async Task<IActionResult> QueryHotelDataInfo(QueryHotelDataInfoDto dto)
  697. {
  698. var _view = await _hotelDataRep._Info(dto.PortType, dto.Id);
  699. if (_view.Code == 0)
  700. {
  701. return Ok(JsonView(true, _view.Msg, _view.Data));
  702. }
  703. return Ok(JsonView(false, _view.Msg));
  704. }
  705. /// <summary>
  706. /// 酒店资料下拉框数据
  707. /// </summary>
  708. /// <returns></returns>
  709. [HttpPost]
  710. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  711. public async Task<IActionResult> QueryHotelDataSelect()
  712. {
  713. try
  714. {
  715. var HotelData = _carDataRep.QueryDto<Res_HotelData, QueryHotelDataSelect>().ToList();
  716. if (HotelData.Count == 0)
  717. {
  718. return Ok(JsonView(false, "暂无数据!"));
  719. }
  720. //解密
  721. foreach (var item in HotelData)
  722. {
  723. EncryptionProcessor.DecryptProperties(item);
  724. }
  725. HotelData.Add(new QueryHotelDataSelect { Id = 0, City = "全部" });
  726. HotelData = HotelData.Where((x, i) => HotelData.FindIndex(z => z.City == x.City && z.City != "") == i).ToList();
  727. HotelData = HotelData.OrderBy(x => x.Id).ToList();
  728. var data = new List<QueryHotelDataSelect>();
  729. foreach (var Hotel in HotelData)
  730. {
  731. if (!string.IsNullOrWhiteSpace(Hotel.City))
  732. {
  733. data.Add(Hotel);
  734. }
  735. }
  736. return Ok(JsonView(true, "查询成功", data));
  737. }
  738. catch (Exception)
  739. {
  740. return Ok(JsonView(false, "程序错误!"));
  741. throw;
  742. }
  743. }
  744. /// <summary>
  745. /// 酒店资料操作(Status:1.新增,2.修改)
  746. /// </summary>
  747. /// <param name="dto"></param>
  748. /// <returns></returns>
  749. [HttpPost]
  750. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  751. public async Task<IActionResult> OperationHotelData(OperationHotelDto dto)
  752. {
  753. if (string.IsNullOrEmpty(dto.City)) return Ok(JsonView(false, "请检查酒店所在城市是否填写!"));
  754. if (string.IsNullOrEmpty(dto.Name)) return Ok(JsonView(false, "请检查酒店名称是否填写!"));
  755. if (string.IsNullOrEmpty(dto.Address)) return Ok(JsonView(false, "请检查酒店地址是否填写正确!"));
  756. if (string.IsNullOrEmpty(dto.Tel)) return Ok(JsonView(false, "请检查酒店联系方式是否填写正确!"));
  757. Result result = await _hotelDataRep.OperationHotelData(dto);
  758. if (result.Code != 0) return Ok(JsonView(false, result.Msg));
  759. return Ok(JsonView(true, result.Msg));
  760. }
  761. /// <summary>
  762. /// 酒店资料操作(删除)
  763. /// </summary>
  764. /// <returns></returns>
  765. [HttpPost]
  766. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  767. public async Task<IActionResult> DelHotelData(DelHotelDataDto dto)
  768. {
  769. try
  770. {
  771. var res = await _hotelDataRep.SoftDeleteByIdAsync<Res_HotelData>(dto.Id.ToString(), dto.DeleteUserId);
  772. if (!res)
  773. {
  774. return Ok(JsonView(false, "删除失败"));
  775. }
  776. return Ok(JsonView(true, "删除成功!"));
  777. }
  778. catch (Exception ex)
  779. {
  780. return Ok(JsonView(false, "程序错误!"));
  781. throw;
  782. }
  783. }
  784. #endregion
  785. #region 签证费用资料
  786. /// <summary>
  787. /// 签证费用资料查询
  788. /// </summary>
  789. /// <param name="dto"></param>
  790. /// <returns></returns>
  791. [HttpPost]
  792. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  793. public async Task<IActionResult> QueryCountryFeeCost(DtoBase dto)
  794. {
  795. var portType = dto.PortType;
  796. if (portType != 1 && portType != 2 && portType != 3)
  797. return Ok(JsonView(false, "请传入PortType参数!1:Web,2:Android,3:IOS"));
  798. var countryVisaFees = await _countryFeeRep
  799. .QueryDto<Res_CountryFeeCost, CountryFeeCostView>()
  800. .OrderByDescending(x => x.CreateTime)
  801. .ToListAsync();
  802. if (countryVisaFees == null || countryVisaFees.Count == 0)
  803. return Ok(JsonView(false, "暂无数据!"));
  804. return Ok(JsonView(true, "查询成功", countryVisaFees));
  805. }
  806. /// <summary>
  807. /// 签证费用资料查询 Page New
  808. /// </summary>
  809. /// <param name="dto"></param>
  810. /// <returns></returns>
  811. [HttpPost]
  812. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  813. public async Task<IActionResult> QueryVisaCountryFeeCosts(QueryVisaCountryFeeCostsDto dto)
  814. {
  815. // 校验端口类型
  816. if (dto.PortType != 1 && dto.PortType != 2 && dto.PortType != 3)
  817. return Ok(JsonView(false, "请传入PortType参数!1:Web,2:Android,3:IOS"));
  818. var portType = dto.PortType;
  819. RefAsync<int> total = 0;
  820. var query = _countryFeeRep._sqlSugar
  821. .Queryable<Res_CountryFeeCost>()
  822. .Where(x => x.IsDel == 0)
  823. .WhereIF(dto.VisaFeeType >= 0, x => x.VisaFeeType == dto.VisaFeeType)
  824. .WhereIF(!string.IsNullOrEmpty(dto.CountryName), x => x.VisaCountry.Contains(dto.CountryName))
  825. .OrderByDescending(x => x.CreateTime);
  826. var countryVisaFees = await query.ToPageListAsync(dto.PageIndex, dto.PageSize, total);
  827. if (!countryVisaFees.Any()) return Ok(JsonView(true, "暂无数据!"));
  828. var view = _mapper.Map<List<CountryFeeCostView>>(countryVisaFees);
  829. return Ok(JsonView(true, "查询成功", view, total));
  830. }
  831. /// <summary>
  832. /// 签证费用资料操作(Status:1.新增,2.修改)
  833. /// </summary>
  834. /// <param name="dto"></param>
  835. /// <returns></returns>
  836. [HttpPost]
  837. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  838. public async Task<IActionResult> OperationCountryFeeCost(OperationCountryFeeCostDto dto)
  839. {
  840. if (string.IsNullOrWhiteSpace(dto.VisaContinent) || string.IsNullOrWhiteSpace(dto.VisaCountry))
  841. return Ok(JsonView(false, "请检查州名和国家名是否填写!"));
  842. return Ok(await _countryFeeRep.OperationCountryFeeCost(dto));
  843. }
  844. /// <summary>
  845. /// 签证费用资料操作(删除)
  846. /// </summary>
  847. /// <returns></returns>
  848. [HttpPost]
  849. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  850. public async Task<IActionResult> DelCountryFeeCost(DelCountryFeeCostDto dto)
  851. {
  852. var res = await _countryFeeRep.SoftDeleteByIdAsync<Res_CountryFeeCost>(dto.Id.ToString(), dto.DeleteUserId);
  853. if (!res) return Ok(JsonView(false, "删除失败"));
  854. return Ok(JsonView(true, "删除成功!"));
  855. }
  856. #endregion
  857. #region 签证费用资料 New
  858. /// <summary>
  859. /// 签证费用标准 Init
  860. /// </summary>
  861. /// <returns></returns>
  862. [HttpPost]
  863. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  864. public async Task<IActionResult> VisaFeeStandardInit()
  865. {
  866. //签证费用类型数据
  867. var visaFeeTypeData = new List<KeyValuePair<int, string>>
  868. {
  869. new(0, "因公"),
  870. new(1, "因私")
  871. };
  872. //签证费用归属省份
  873. var data = new
  874. {
  875. VisaFeeType = visaFeeTypeData.Select(x => new { Id = x.Key, Text = x.Value }),
  876. Province = _provinceData.Select(x => new { Id = x.Key, Text = x.Value })
  877. };
  878. return Ok(JsonView(data));
  879. }
  880. /// <summary>
  881. /// 签证费用标准 Info
  882. /// </summary>
  883. /// <param name="dto"></param>
  884. /// <returns></returns>
  885. [HttpPost]
  886. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  887. public async Task<IActionResult> VisaFeeStandardInfo(VisaFeeStandardInfoDto dto)
  888. {
  889. var portType = dto.PortType;
  890. if (portType != 1 && portType != 2 && portType != 3)
  891. return Ok(JsonView(false, "请传入PortType参数!1:Web,2:Android,3:IOS"));
  892. return Ok(await _countryFeeRep.InfoAsync(dto.Id));
  893. }
  894. /// <summary>
  895. /// 签证费用标准 List
  896. /// </summary>
  897. /// <param name="dto"></param>
  898. /// <returns></returns>
  899. [HttpPost]
  900. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  901. public async Task<IActionResult> VisaFeeStandardList(VisaFeeStandardListDto dto)
  902. {
  903. return Ok(await _countryFeeRep.PageListAsync(dto));
  904. }
  905. /// <summary>
  906. /// 签证费用标准 Save
  907. /// </summary>
  908. /// <param name="dto"></param>
  909. /// <returns></returns>
  910. [HttpPost]
  911. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  912. public async Task<IActionResult> VisaFeeStandardSave(VisaFeeStandardSaveDto dto)
  913. {
  914. return Ok(await _countryFeeRep.SaveAsync(dto));
  915. }
  916. /// <summary>
  917. /// 签证费用标准 SoftDel
  918. /// </summary>
  919. /// <param name="dto"></param>
  920. /// <returns></returns>
  921. [HttpPost]
  922. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  923. public async Task<IActionResult> VisaFeeStandardSoftDel(VisaFeeStandardSoftDelDto dto)
  924. {
  925. return Ok(await _countryFeeRep.SoftDelAsync(dto.CurrUserId, dto.Id));
  926. }
  927. /// <summary>
  928. /// 根据省份获取签证费用报价
  929. /// </summary>
  930. /// <param name="dto"></param>
  931. /// <returns></returns>
  932. [HttpPost]
  933. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  934. public async Task<IActionResult> VisaFeeQuoteByProvince(VisaFeesByProvinceDto dto)
  935. {
  936. //省份验证
  937. var provinceIds = _provinceData.Select(x => x.Key).ToList();
  938. if (!provinceIds.Contains(dto.ProvinceId))
  939. {
  940. return Ok(JsonView(false, "请选择有效的省份!"));
  941. }
  942. return Ok(await _countryFeeRep.VisaFeeQuoteAsync(dto.ProvinceId, dto.CountryName));
  943. }
  944. #endregion
  945. #region 物料信息、供应商维护
  946. #region 供应商
  947. /// <summary>
  948. /// 物料供应商查询
  949. /// </summary>
  950. /// <param name="_jsonDto">Json序列化</param>
  951. /// <returns></returns>
  952. [HttpPost]
  953. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  954. public async Task<IActionResult> PostSearchItemVendor(JsonDtoBase _jsonDto)
  955. {
  956. if (string.IsNullOrEmpty(_jsonDto.Paras))
  957. {
  958. return Ok(JsonView(false, "参数为空"));
  959. }
  960. Search_ResItemVendorDto _ItemVendorDto = System.Text.Json.JsonSerializer.Deserialize<Search_ResItemVendorDto>(_jsonDto.Paras);
  961. if (_ItemVendorDto != null)
  962. {
  963. if (_ItemVendorDto.SearchType == 2) //获取列表
  964. {
  965. Res_ItemVendorListView rstList = _resItemInfoRep.GetVendorList(_ItemVendorDto);
  966. return Ok(rstList);
  967. }
  968. else
  969. {
  970. Res_ItemVendorView rstInfo = _resItemInfoRep.getVendorInfo(_ItemVendorDto);
  971. return Ok(rstInfo);
  972. }
  973. }
  974. else
  975. {
  976. return Ok(JsonView(false, "参数反序列化失败"));
  977. }
  978. //return Ok(JsonView(false));
  979. }
  980. /// <summary>
  981. /// 创建/编辑/删除供应商信息
  982. /// </summary>
  983. /// <param name="_dto"></param>
  984. /// <returns></returns>
  985. [HttpPost]
  986. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  987. public async Task<IActionResult> PostEditItemVendor(Edit_ResItemVendorDto _dto)
  988. {
  989. bool rst = false;
  990. if (_dto.EditType >= 0)
  991. {
  992. if (string.IsNullOrEmpty(_dto.VendorFullName))
  993. {
  994. return Ok(JsonView(false, "全称未填写"));
  995. }
  996. if (string.IsNullOrEmpty(_dto.VendorLinker))
  997. {
  998. return Ok(JsonView(false, "联系人未填写"));
  999. }
  1000. if (string.IsNullOrEmpty(_dto.VendorMobile))
  1001. {
  1002. return Ok(JsonView(false, "联系人手机号未填写"));
  1003. }
  1004. if (string.IsNullOrEmpty(_dto.BusRange))
  1005. {
  1006. return Ok(JsonView(false, "经营范围未选择"));
  1007. }
  1008. if (_dto.EditType == 0)
  1009. {
  1010. var checkEmpty = _resItemInfoRep.Query<Res_ItemVendor>(s => s.FullName == _dto.VendorFullName).First();
  1011. if (checkEmpty != null)
  1012. {
  1013. return Ok(JsonView(false, "已存在同名供应商"));
  1014. }
  1015. rst = await _resItemInfoRep.addVendorInfo(_dto);
  1016. }
  1017. else if (_dto.EditType == 1)
  1018. {
  1019. if (_dto.VendorId > 0)
  1020. {
  1021. Res_ItemVendor _entity = _mapper.Map<Res_ItemVendor>(_dto);
  1022. rst = await _resItemInfoRep.updVendorInfo(_entity);
  1023. }
  1024. else
  1025. {
  1026. return Ok(JsonView(false, "供应商不存在"));
  1027. }
  1028. }
  1029. }
  1030. else
  1031. {
  1032. if (_dto.VendorId < 1 || _dto.SysUserId < 1)
  1033. {
  1034. return Ok(JsonView(false, "用户Id或供应商Id不存在"));
  1035. }
  1036. Res_ItemVendor _entity = _mapper.Map<Res_ItemVendor>(_dto);
  1037. rst = await _resItemInfoRep.delVendorInfo(_entity);
  1038. }
  1039. return Ok(JsonView(rst));
  1040. }
  1041. #endregion
  1042. #region 物料信息
  1043. /// <summary>
  1044. /// 物料信息查询
  1045. /// </summary>
  1046. /// <param name="paras">Json序列化</param>
  1047. /// <returns></returns>
  1048. [HttpPost]
  1049. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  1050. public async Task<IActionResult> PostSearchItemInfo(string paras)
  1051. {
  1052. if (string.IsNullOrEmpty(paras))
  1053. {
  1054. return Ok(JsonView(false, "参数为空"));
  1055. }
  1056. Search_ItemInfoDto _ItemInfoDto = System.Text.Json.JsonSerializer.Deserialize<Search_ItemInfoDto>(paras);
  1057. if (_ItemInfoDto != null)
  1058. {
  1059. if (_ItemInfoDto.SearchType == 2) //获取列表
  1060. {
  1061. Res_ItemInfoListView rstList = _resItemInfoRep.GetItemList(_ItemInfoDto);
  1062. return Ok(rstList);
  1063. }
  1064. else
  1065. {
  1066. Res_ItemInfoView rstInfo = _resItemInfoRep.getItemInfo(_ItemInfoDto);
  1067. return Ok(rstInfo);
  1068. }
  1069. }
  1070. else
  1071. {
  1072. return Ok(JsonView(false, "参数反序列化失败"));
  1073. }
  1074. //return Ok(JsonView(false));
  1075. }
  1076. /// <summary>
  1077. /// 创建/编辑/删除物料信息
  1078. /// </summary>
  1079. ///
  1080. /// <returns></returns>
  1081. [HttpPost]
  1082. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  1083. public async Task<IActionResult> PostEditItemInfo(Edit_ResItemInfoDto _dto)
  1084. {
  1085. bool rst = false;
  1086. if (_dto.EditType >= 0)
  1087. {
  1088. if (_dto.VendorId < 1)
  1089. {
  1090. return Ok(JsonView(false, "未选择供应商"));
  1091. }
  1092. if (string.IsNullOrEmpty(_dto.ItemName))
  1093. {
  1094. return Ok(JsonView(false, "物料名称为空"));
  1095. }
  1096. if (_dto.ItemTypeId < 1)
  1097. {
  1098. return Ok(JsonView(false, "未选择物料类型"));
  1099. }
  1100. if (_dto.SysUserId < 1)
  1101. {
  1102. return Ok(JsonView(false, "当前操作用户Id为空"));
  1103. }
  1104. if (_dto.CurrRate <= 0)
  1105. {
  1106. return Ok(JsonView(false, "物料录入价格不能小于等于0"));
  1107. }
  1108. if (_dto.EditType == 0)
  1109. {
  1110. //判断物料名称、类型、供应商全部重复
  1111. var checkEmpty = _resItemInfoRep.Query<Res_ItemDetailInfo>(s => s.ItemName == _dto.ItemName && s.ItemTypeId == _dto.ItemTypeId && s.VendorId == _dto.VendorId).First();
  1112. if (checkEmpty != null)
  1113. {
  1114. return Ok(JsonView(false, "已存在重复物料信息"));
  1115. }
  1116. Res_ItemDetailInfo _entity = _mapper.Map<Res_ItemDetailInfo>(_dto);
  1117. DateTime dtNow = DateTime.Now;
  1118. _entity.CreateUserId = _dto.SysUserId;
  1119. _entity.IsDel = 0;
  1120. _entity.MaxRate = _dto.CurrRate;
  1121. _entity.MaxDt = dtNow;
  1122. _entity.CurrRate = _dto.CurrRate;
  1123. _entity.CurrDt = dtNow;
  1124. _entity.MinRate = _dto.CurrRate;
  1125. _entity.MinDt = dtNow;
  1126. rst = await _resItemInfoRep.AddAsync<Res_ItemDetailInfo>(_entity) > 0;
  1127. }
  1128. else if (_dto.EditType == 1)
  1129. {
  1130. if (_dto.ItemId > 0)
  1131. {
  1132. Res_ItemDetailInfo _entity = _mapper.Map<Res_ItemDetailInfo>(_dto);
  1133. _entity.Id = _dto.ItemId;
  1134. rst = await _resItemInfoRep.updItemInfo(_entity);
  1135. }
  1136. else
  1137. {
  1138. return Ok(JsonView(false, "供应商不存在"));
  1139. }
  1140. }
  1141. }
  1142. else
  1143. {
  1144. if (_dto.ItemId < 1 || _dto.SysUserId < 1)
  1145. {
  1146. return Ok(JsonView(false, "用户Id或物料信息Id不存在"));
  1147. }
  1148. Res_ItemDetailInfo _entity = _mapper.Map<Res_ItemDetailInfo>(_dto);
  1149. rst = await _resItemInfoRep.delItemInfo(_entity);
  1150. }
  1151. return Ok(JsonView(rst));
  1152. }
  1153. #endregion
  1154. #region 物料类型获取
  1155. #endregion
  1156. #endregion
  1157. #region 备忘录
  1158. /// <summary>
  1159. /// 备忘录查询
  1160. /// </summary>
  1161. /// <param name="_jsonDto">Json序列化</param>
  1162. /// <returns></returns>
  1163. [HttpPost]
  1164. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  1165. public async Task<IActionResult> PostSearchMemo(JsonDtoBase _jsonDto)
  1166. {
  1167. if (string.IsNullOrEmpty(_jsonDto.Paras))
  1168. {
  1169. return Ok(JsonView(false, "参数为空"));
  1170. }
  1171. Search_ResMemoDto _memoDto = JsonConvert.DeserializeObject<Search_ResMemoDto>(_jsonDto.Paras);
  1172. if (_memoDto != null)
  1173. {
  1174. if (_memoDto.SearchType == 2)
  1175. {
  1176. //获取列表
  1177. string sqlWhere = string.Format(" Where IsDel=0 ");
  1178. #region SqlWhere
  1179. if (!string.IsNullOrEmpty(_memoDto.Abstracts))
  1180. {
  1181. sqlWhere += string.Format(@" And m.Abstracts Like '%{0}%' ", _memoDto.Abstracts);
  1182. }
  1183. if (!string.IsNullOrEmpty(_memoDto.Title))
  1184. {
  1185. sqlWhere += string.Format(@" And m.Title Like '%{0}%' ", _memoDto.Title);
  1186. }
  1187. if (_memoDto.ReadLevel > 0)
  1188. {
  1189. sqlWhere += string.Format(@" And m.ReadLevel = {0} ", _memoDto.ReadLevel);
  1190. }
  1191. #endregion
  1192. int currPIndex = (((_memoDto.PageIndex > 0) ? (_memoDto.PageIndex - 1) : 0) * _memoDto.PageSize) + 1;
  1193. int currPSize = (((_memoDto.PageIndex > 0) ? (_memoDto.PageIndex - 1) : 0) + 1) * _memoDto.PageSize;
  1194. string sql = string.Format(@" Select * From(Select ROW_NUMBER() Over(order By m.Id desc) as RowNumber,
  1195. m.Id as MemoId,d.DepName as DepartmentName,m.ReadLevel,m.Title,m.Abstracts,
  1196. m.LastedEditDt,m.LastedEditorId
  1197. From Res_Memo as m With(Nolock) Inner Join Sys_Users as u With(Nolock) On m.CreateUserId=u.Id
  1198. Inner Join Sys_Department as d With(Nolock) On u.DepId=d.Id {2}
  1199. ) as tb Where tb.RowNumber Between {0} And {1} ", currPIndex, currPSize, sqlWhere);
  1200. Res_MemoListView rst = new Res_MemoListView();
  1201. rst.CurrPageIndex = currPIndex;
  1202. rst.CurrPageSize = currPSize;
  1203. List<Res_MemoView> dataSource = _carDataRep._sqlSugar.SqlQueryable<Res_MemoView>(sql).ToList();
  1204. Dictionary<int, string> userDic = new Dictionary<int, string>();
  1205. foreach (var item in dataSource)
  1206. {
  1207. if (userDic.ContainsKey(item.LastedEditorId))
  1208. {
  1209. item.LastedEditor = userDic[item.LastedEditorId];
  1210. }
  1211. else
  1212. {
  1213. var _sysUser = _carDataRep.Query<Sys_Users>(s => s.Id == item.LastedEditorId).First();
  1214. userDic[item.LastedEditorId] = _sysUser.CnName;
  1215. item.LastedEditor = _sysUser.CnName;
  1216. }
  1217. }
  1218. rst.DataList = new List<Res_MemoView>(dataSource);
  1219. if (rst.DataList.Count > 0)
  1220. {
  1221. string sqlCount = string.Format(@" Select Id From Res_Memo as m With(Nolock) {0} ", sqlWhere);
  1222. int dataCount = _carDataRep._sqlSugar.SqlQueryable<Res_MemoInfo>(sqlCount).Count();
  1223. rst.DataCount = dataCount;
  1224. }
  1225. return Ok(JsonView(rst));
  1226. }
  1227. else
  1228. {
  1229. //获取对象
  1230. string sqlSingle = string.Format(@" Select
  1231. m.Id as MemoId,d.DepName as DepartmentName,m.ReadLevel,m.Title,m.Abstracts,
  1232. m.LastedEditDt,m.LastedEditor,m.MDFilePath
  1233. From Res_Memo as m With(Nolock) Inner Join Sys_Users as u With(Nolock) On m.CreateUserId=u.Id
  1234. Inner Join Sys_Department as d With(Nolock) On u.DepId=d.Id Where m.Id={0} ", _memoDto.MemoId);
  1235. Res_MemoView _result = _carDataRep._sqlSugar.SqlQueryable<Res_MemoView>(sqlSingle).First();
  1236. if (_result != null)
  1237. {
  1238. var _sysUser = _carDataRep.Query<Sys_Users>(s => s.Id == _result.LastedEditorId).First();
  1239. _result.LastedEditor = _sysUser.CnName;
  1240. _result.MarkDownContent = new IOOperatorHelper().Read(_result.MDFilePath);
  1241. return Ok(JsonView(_result));
  1242. }
  1243. }
  1244. }
  1245. else
  1246. {
  1247. return Ok(JsonView(false, "参数反序列化失败"));
  1248. }
  1249. return Ok(JsonView(false));
  1250. }
  1251. /// <summary>
  1252. /// 创建/编辑/删除备忘录信息
  1253. /// </summary>
  1254. /// <param name="_dto"></param>
  1255. /// <returns></returns>
  1256. [HttpPost]
  1257. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  1258. public async Task<IActionResult> PostEditMemo(Edit_ResMemoDto _dto)
  1259. {
  1260. bool rst = false;
  1261. if (_dto.SysUserId < 1)
  1262. {
  1263. return Ok(JsonView(false, "操作人失效"));
  1264. }
  1265. var _sysUser = _resItemInfoRep.Query<Sys_Users>(s => s.Id == _dto.SysUserId).First();
  1266. if (_sysUser == null)
  1267. {
  1268. return Ok(JsonView(false, "操作人失效02"));
  1269. }
  1270. if (_dto.EditType >= 0)
  1271. {
  1272. if (_dto.ReadLevel < 1)
  1273. {
  1274. return Ok(JsonView(false, "未知的阅读等级"));
  1275. }
  1276. if (string.IsNullOrEmpty(_dto.Title.Trim()))
  1277. {
  1278. return Ok(JsonView(false, "标题不能为空"));
  1279. }
  1280. if (string.IsNullOrEmpty(_dto.Content.Trim()))
  1281. {
  1282. return Ok(JsonView(false, "正文内容不能为空"));
  1283. }
  1284. //新增备忘录
  1285. DateTime dtNow = DateTime.Now;
  1286. string dir = AppSettingsHelper.Get("MemoCurrPath");
  1287. string fileName = dtNow.ToString("yyyyMMddHHmmss") + _dto.Title + ".md";
  1288. string content = JsonConvert.SerializeObject(_dto.Content);
  1289. if (_dto.EditType == 0)//新增
  1290. {
  1291. string savePath = new IOOperatorHelper().Write_CoverFile(content, dir, fileName);
  1292. if (savePath.Length > 0)
  1293. {
  1294. Res_MemoInfo _insert = new Res_MemoInfo();
  1295. _insert.Abstracts = _dto.Abstracts;
  1296. _insert.Title = _dto.Title;
  1297. _insert.DepartmentId = _sysUser.DepId;
  1298. _insert.CreateUserId = _sysUser.Id;
  1299. _insert.LastedEditDt = dtNow;
  1300. _insert.LastedEditor = _sysUser.Id;
  1301. _insert.MDFilePath = savePath;
  1302. _insert.ReadLevel = _dto.ReadLevel;
  1303. _insert.Title = _dto.Title;
  1304. int result = await _resItemInfoRep.AddAsync(_insert);
  1305. return Ok(JsonView(result > 0));
  1306. }
  1307. else
  1308. {
  1309. return Ok(JsonView(false, "路径保存失败"));
  1310. }
  1311. }
  1312. else//修改
  1313. {
  1314. if (_dto.MemoId < 1)
  1315. {
  1316. return Ok(JsonView(false, "MemoId不存在"));
  1317. }
  1318. Res_MemoInfo _source = _resItemInfoRep.Query<Res_MemoInfo>(s => s.Id == _dto.MemoId).First();
  1319. if (_source == null)
  1320. {
  1321. return Ok(JsonView(false, "MemoInfo不存在"));
  1322. }
  1323. //修改
  1324. string sourcePath = _source.MDFilePath;
  1325. string recycDir = AppSettingsHelper.Get("MemoRecycleBinPath");
  1326. new IOOperatorHelper().MoveFile(sourcePath, recycDir);
  1327. string savePath = new IOOperatorHelper().Write_CoverFile(content, dir, fileName);
  1328. if (savePath.Length > 0)
  1329. {
  1330. var result = await _resItemInfoRep._sqlSugar.Updateable<Res_MemoInfo>()
  1331. .SetColumns(it => it.LastedEditDt == DateTime.Now)
  1332. .SetColumns(it => it.LastedEditor == _sysUser.Id)
  1333. .SetColumns(it => it.Abstracts == _dto.Abstracts)
  1334. .SetColumns(it => it.MDFilePath == savePath)
  1335. .SetColumns(it => it.ReadLevel == _dto.ReadLevel)
  1336. .SetColumns(it => it.Title == _dto.Title)
  1337. .Where(s => s.Id == _source.Id)
  1338. .ExecuteCommandAsync();
  1339. return Ok(JsonView(result > 0));
  1340. }
  1341. }
  1342. }
  1343. else
  1344. {
  1345. //删除
  1346. if (_dto.MemoId < 1)
  1347. {
  1348. return Ok(JsonView(false, "MemoId不存在"));
  1349. }
  1350. Res_MemoInfo _source = _resItemInfoRep.Query<Res_MemoInfo>(s => s.Id == _dto.MemoId).First();
  1351. if (_source == null)
  1352. {
  1353. return Ok(JsonView(false, "MemoInfo不存在"));
  1354. }
  1355. //修改
  1356. string sourcePath = _source.MDFilePath;
  1357. string recycDir = AppSettingsHelper.Get("MemoRecycleBinPath");
  1358. new IOOperatorHelper().MoveFile(sourcePath, recycDir);
  1359. var result = await _resItemInfoRep._sqlSugar.Updateable<Res_MemoInfo>()
  1360. .SetColumns(it => it.IsDel == 1)
  1361. .SetColumns(it => it.DeleteUserId == _sysUser.Id)
  1362. .SetColumns(it => it.DeleteTime == DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"))
  1363. .Where(s => s.Id == _source.Id)
  1364. .ExecuteCommandAsync();
  1365. return Ok(JsonView(result > 0));
  1366. }
  1367. return Ok(JsonView(rst));
  1368. }
  1369. #endregion
  1370. #region 商邀资料
  1371. /// <summary>
  1372. /// 商邀资料 基础数据源
  1373. /// </summary>
  1374. /// <param name="dto"></param>
  1375. /// <returns></returns>
  1376. [HttpPost]
  1377. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  1378. public async Task<IActionResult> QueryIOAInitData(QueryIOAInitDataDto dto)
  1379. {
  1380. #region 参数验证
  1381. if (dto.PortType < 1) return Ok(JsonView(false, "请传入有效的PortType参数!"));
  1382. #endregion
  1383. var ExcludedKeyStr = new string[] { "快递费" };
  1384. var ClientKeyStr = new string[] { "邀请函翻译" };
  1385. List<Grp_DelegationInfo> _DelegationInfos = _sqlSugar.Queryable<Grp_DelegationInfo>().Where(it => it.IsDel == 0).OrderByDescending(it => it.JietuanTime).ToList();
  1386. List<Res_InvitationOfficialActivityData> _ioaDatas = _sqlSugar.Queryable<Res_InvitationOfficialActivityData>()
  1387. .Where(it => it.IsDel == 0)
  1388. .Select(x => new Res_InvitationOfficialActivityData
  1389. {
  1390. Country = x.Country,
  1391. UnitName = x.UnitName,
  1392. Contact = x.Contact,
  1393. Field = x.Field,
  1394. })
  1395. .ToList();
  1396. foreach (var item in _ioaDatas)
  1397. {
  1398. EncryptionProcessor.DecryptProperties(item);
  1399. }
  1400. foreach (var item in ExcludedKeyStr)
  1401. {
  1402. _ioaDatas = _ioaDatas.Where(x => string.IsNullOrWhiteSpace(x.UnitName) || !x.UnitName.Contains(item)).ToList();
  1403. }
  1404. foreach (var item in ClientKeyStr)
  1405. {
  1406. _ioaDatas = _ioaDatas.Where(x => string.IsNullOrWhiteSpace(x.Contact) || !x.Contact.Contains(item)).ToList();
  1407. }
  1408. List<Sys_Users> _Users = _sqlSugar.Queryable<Sys_Users>().Where(it => it.IsDel == 0).ToList();
  1409. var _countryData = _ioaDatas.Select(it => it.Country).Distinct().ToList(); _countryData.Remove("");
  1410. var _inviterData = _ioaDatas.Select(it => it.UnitName).Distinct().ToList(); _inviterData.Remove("");
  1411. var _contactData = _ioaDatas.Select(it => it.Contact).Distinct().ToList(); _contactData.Remove("");
  1412. var _domainData = _ioaDatas.Select(it => it.Field).Distinct().ToList(); _domainData.Remove("");
  1413. var _groupNameData = _DelegationInfos.Select(it => new { it.Id, it.TeamName }).ToList();
  1414. var _userNameData = _Users.Select(it => new { it.Id, it.CnName }).ToList();
  1415. var _data = new
  1416. {
  1417. CountryData = _countryData,
  1418. InviterData = _inviterData,
  1419. DomainData = _domainData,
  1420. ContactData = _contactData,
  1421. GroupNameData = _groupNameData,
  1422. UserNameData = _userNameData,
  1423. };
  1424. return Ok(JsonView(true, $"查询成功!", _data));
  1425. }
  1426. /// <summary>
  1427. /// 商邀资料查询
  1428. /// </summary>
  1429. /// <param name="dto"></param>
  1430. /// <returns></returns>
  1431. [HttpPost]
  1432. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  1433. public async Task<IActionResult> QueryInvitationOfficialActivityData(QueryInvitationOfficialActivityDataDto dto)
  1434. {
  1435. var ExcludedKeyStr = new string[] { "快递费" };
  1436. var ClientKeyStr = new string[] { "邀请函翻译" };
  1437. var columns = "Id,UnitName,Contact,City";
  1438. try
  1439. {
  1440. #region 参数验证
  1441. if (dto.PageIndex < 1) return Ok(JsonView(false, "请传入有效的PageIndex参数!"));
  1442. if (dto.PageSize < 1) return Ok(JsonView(false, "请传入有效的PageSize参数!"));
  1443. #endregion
  1444. //string sqlWhere = string.Empty;
  1445. if (!string.IsNullOrWhiteSpace(dto.Country)) { columns += ",Country"; }
  1446. //if (!string.IsNullOrWhiteSpace(dto.UnitName)) { columns += ",UnitName"; }
  1447. //if (!string.IsNullOrWhiteSpace(dto.Contact)) { columns += ",Contact"; }
  1448. if (!string.IsNullOrWhiteSpace(dto.Delegation)) { columns += ",Delegation"; }
  1449. if (!string.IsNullOrWhiteSpace(dto.Field)) { columns += ",Field"; }
  1450. //if (dto.CreateUserId != 0 && !string.IsNullOrWhiteSpace(dto.CreateUserId.ToString())) { columns += ",CreateUserId"; }
  1451. //if (!string.IsNullOrWhiteSpace(dto.StartCreateTime) && !string.IsNullOrWhiteSpace(dto.EndCreateTime))
  1452. //{
  1453. // columns += ",CreateUserId";
  1454. //}
  1455. DateTime endTime = new DateTime();
  1456. var InvitationOfficialActivityDataList = _sqlSugar.Queryable<Res_InvitationOfficialActivityData>()
  1457. .Where(x => x.IsDel == 0)
  1458. .WhereIF(dto.CreateUserId != 0 && !string.IsNullOrWhiteSpace(dto.CreateUserId.ToString()), x => x.CreateUserId == dto.CreateUserId)
  1459. .WhereIF(DateTime.TryParse(dto.StartCreateTime, out DateTime startTime) && DateTime.TryParse(dto.EndCreateTime, out endTime), x => x.CreateTime >= startTime && x.CreateTime <= endTime)
  1460. .Select(columns)
  1461. .ToList();
  1462. foreach (var item in InvitationOfficialActivityDataList)
  1463. {
  1464. EncryptionProcessor.DecryptProperties(item);
  1465. }
  1466. InvitationOfficialActivityDataList = InvitationOfficialActivityDataList
  1467. .WhereIF(!string.IsNullOrWhiteSpace(dto.Country), x => (!string.IsNullOrWhiteSpace(x.Country) && x.Country.Contains(dto.Country)) || (!string.IsNullOrWhiteSpace(x.City) && x.City.Contains(dto.Country)))
  1468. .WhereIF(!string.IsNullOrWhiteSpace(dto.UnitName), x => !string.IsNullOrWhiteSpace(x.UnitName) && x.UnitName.Contains(dto.UnitName))
  1469. .WhereIF(!string.IsNullOrWhiteSpace(dto.Contact), x => !string.IsNullOrWhiteSpace(x.Contact) && x.Contact.Contains(dto.Contact))
  1470. .WhereIF(!string.IsNullOrWhiteSpace(dto.Delegation), x => !string.IsNullOrWhiteSpace(x.Delegation) && x.Delegation.Contains(dto.Delegation))
  1471. .WhereIF(!string.IsNullOrWhiteSpace(dto.Field), x => !string.IsNullOrWhiteSpace(x.Field) && x.Field.Contains(dto.Field))
  1472. .ToList();
  1473. foreach (var item in ExcludedKeyStr)
  1474. {
  1475. InvitationOfficialActivityDataList = InvitationOfficialActivityDataList.Where(x => string.IsNullOrWhiteSpace(x.UnitName) || !x.UnitName.Contains(item)).ToList();
  1476. //sqlWhere += $" And i.UnitName not like '%{item}%' ";
  1477. }
  1478. foreach (var item in ClientKeyStr)
  1479. {
  1480. InvitationOfficialActivityDataList = InvitationOfficialActivityDataList.Where(x => string.IsNullOrWhiteSpace(x.Contact) || !x.Contact.Contains(item)).ToList();
  1481. //sqlWhere += $" And i.Contact not like '%{item}%' ";
  1482. }
  1483. var ids = InvitationOfficialActivityDataList.Select(x => x.Id);
  1484. RefAsync<int> totalCount = 0;
  1485. var _ivitiesViews = await _sqlSugar.Queryable<Res_InvitationOfficialActivityData>()
  1486. .LeftJoin<Sys_Users>((a, b) => b.IsDel == 0 && a.CreateUserId == b.Id)
  1487. .Where((a, b) => ids.Contains(a.Id))
  1488. .OrderByDescending((a, b) => a.CreateTime)
  1489. .Select((a, b) => new InvitationOfficialActivityDataView
  1490. {
  1491. FaceBook = a.FaceBook,
  1492. Id = a.Id,
  1493. CreateUserId = a.CreateUserId,
  1494. IsDel = 0,
  1495. Address = a.Address,
  1496. Background = a.Background,
  1497. City = a.City,
  1498. Contact = a.Contact,
  1499. Country = a.Country,
  1500. CreateTime = a.CreateTime,
  1501. CreateUserName = b.CnName,
  1502. Delegation = a.Delegation,
  1503. Email = a.Email,
  1504. Fax = a.Fax,
  1505. Field = a.Field,
  1506. FilePath = a.FilePath,
  1507. Ins = a.Ins,
  1508. WeChat = a.WeChat,
  1509. UnitWeb = a.UnitWeb,
  1510. UnitName = a.UnitName,
  1511. UnitInfo = a.UnitInfo,
  1512. Tel = a.Tel,
  1513. SndFilePath = a.SndFilePath,
  1514. SndFileName = a.SndFileName,
  1515. Remark = a.Remark,
  1516. OtherInfo = a.OtherInfo,
  1517. Job = a.Job
  1518. })
  1519. .ToPageListAsync(dto.PageIndex, dto.PageSize, totalCount);
  1520. var allGroupIds = new HashSet<int>();
  1521. foreach (var item in _ivitiesViews)
  1522. {
  1523. EncryptionProcessor.DecryptProperties(item);
  1524. if (!string.IsNullOrEmpty(item.Delegation))
  1525. {
  1526. allGroupIds.UnionWith(item.Delegation.Split(',').Select(x =>
  1527. {
  1528. int id;
  1529. if (int.TryParse(x, out id)) return id;
  1530. return 0;
  1531. }).Where(id => id != 0));
  1532. }
  1533. }
  1534. var _DelegationInfos = _sqlSugar.Queryable<Grp_DelegationInfo>()
  1535. .Where(x => allGroupIds.Contains(x.Id) && x.IsDel == 0)
  1536. .ToList()
  1537. .GroupBy(x => x.Id)
  1538. .ToDictionary(group => group.Key, group => group.Select(g => g.TeamName));
  1539. foreach (var item in _ivitiesViews)
  1540. {
  1541. string groupNameStr = "";
  1542. if (!string.IsNullOrEmpty(item.Delegation))
  1543. {
  1544. var groupIds = item.Delegation.Split(',').Select(x =>
  1545. {
  1546. int id;
  1547. if (int.TryParse(x, out id)) return id;
  1548. return 0;
  1549. })
  1550. .Where(id => id != 0)
  1551. .ToArray();
  1552. foreach (var id in groupIds)
  1553. {
  1554. if (_DelegationInfos.ContainsKey(id))
  1555. {
  1556. groupNameStr += string.Join("", _DelegationInfos[id]) + ",";
  1557. }
  1558. }
  1559. if (groupNameStr.Length > 1)
  1560. {
  1561. groupNameStr = groupNameStr.Substring(0, groupNameStr.Length - 1);
  1562. }
  1563. }
  1564. item.DelegationStr = groupNameStr;
  1565. }
  1566. return Ok(JsonView(true, $"查询成功!", _ivitiesViews, totalCount));
  1567. }
  1568. catch (Exception ex)
  1569. {
  1570. return Ok(JsonView(false, ex.Message));
  1571. }
  1572. }
  1573. /// <summary>
  1574. /// 商邀资料查询-欧洲
  1575. /// </summary>
  1576. /// <returns></returns>
  1577. [HttpPost]
  1578. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  1579. public async Task<IActionResult> QueryInvitationOfficialActivityData1()
  1580. {
  1581. try
  1582. {
  1583. // 定义欧洲国家数组
  1584. string[] europeanCountries = {
  1585. // 北欧国家
  1586. "瑞典", "芬兰", "挪威", "丹麦", "冰岛",
  1587. // 西欧国家
  1588. "英国", "爱尔兰", "法国", "荷兰", "比利时", "卢森堡", "摩纳哥",
  1589. // 中欧国家
  1590. "德国", "瑞士", "奥地利", "波兰", "捷克", "斯洛伐克", "匈牙利", "列支敦士登",
  1591. // 南欧国家
  1592. "意大利", "西班牙", "葡萄牙", "希腊", "马耳他", "圣马力诺", "梵蒂冈", "安道尔",
  1593. "克罗地亚", "斯洛文尼亚", "波斯尼亚和黑塞哥维那", "黑山", "塞尔维亚", "北马其顿", "阿尔巴尼亚", "保加利亚", "罗马尼亚",
  1594. // 东欧国家
  1595. "俄罗斯", "乌克兰", "白俄罗斯", "摩尔多瓦", "爱沙尼亚", "拉脱维亚", "立陶宛"
  1596. };
  1597. var InvitationOfficialActivityDataList = _sqlSugar.Queryable<Res_InvitationOfficialActivityData>()
  1598. .Where(x => x.IsDel == 0)
  1599. .Select(x => new Res_InvitationOfficialActivityData() { Id = x.Id, Country = x.Country })
  1600. .ToList();
  1601. foreach (var item in InvitationOfficialActivityDataList)
  1602. {
  1603. EncryptionProcessor.DecryptProperties(item);
  1604. }
  1605. var ids = InvitationOfficialActivityDataList
  1606. .WhereIF(europeanCountries.Any(), x => !string.IsNullOrWhiteSpace(x.Country) && europeanCountries.Contains(x.Country))
  1607. .Select(x => x.Id)
  1608. .ToList();
  1609. RefAsync<int> totalCount = 0;
  1610. var _ivitiesViews = await _sqlSugar.Queryable<Res_InvitationOfficialActivityData>()
  1611. .LeftJoin<Sys_Users>((a, b) => b.IsDel == 0 && a.CreateUserId == b.Id)
  1612. .Where((a, b) => ids.Contains(a.Id))
  1613. .OrderByDescending((a, b) => a.CreateTime)
  1614. .Select((a, b) => new InvitationOfficialActivityDataView1
  1615. {
  1616. Country = a.Country,
  1617. City = a.City,
  1618. UnitName = a.UnitName,
  1619. Field = a.Field,
  1620. Address = a.Address,
  1621. UnitInfo = a.UnitInfo,
  1622. Delegation = a.Delegation,
  1623. OtherInfo = a.OtherInfo
  1624. })
  1625. .ToListAsync();
  1626. var allGroupIds = new HashSet<int>();
  1627. foreach (var item in _ivitiesViews)
  1628. {
  1629. EncryptionProcessor.DecryptProperties(item);
  1630. if (!string.IsNullOrEmpty(item.Delegation))
  1631. {
  1632. allGroupIds.UnionWith(item.Delegation.Split(',').Select(x =>
  1633. {
  1634. int id;
  1635. if (int.TryParse(x, out id)) return id;
  1636. return 0;
  1637. }).Where(id => id != 0));
  1638. }
  1639. }
  1640. var _DelegationInfos = _sqlSugar.Queryable<Grp_DelegationInfo>()
  1641. .Where(x => allGroupIds.Contains(x.Id) && x.IsDel == 0)
  1642. .ToList()
  1643. .GroupBy(x => x.Id)
  1644. .ToDictionary(group => group.Key, group => group.Select(g => g.TeamName));
  1645. foreach (var item in _ivitiesViews)
  1646. {
  1647. string groupNameStr = "";
  1648. if (!string.IsNullOrEmpty(item.Delegation))
  1649. {
  1650. var groupIds = item.Delegation.Split(',').Select(x =>
  1651. {
  1652. int id;
  1653. if (int.TryParse(x, out id)) return id;
  1654. return 0;
  1655. })
  1656. .Where(id => id != 0)
  1657. .ToArray();
  1658. foreach (var id in groupIds)
  1659. {
  1660. if (_DelegationInfos.ContainsKey(id))
  1661. {
  1662. groupNameStr += string.Join("", _DelegationInfos[id]) + ",";
  1663. }
  1664. }
  1665. if (groupNameStr.Length > 1)
  1666. {
  1667. groupNameStr = groupNameStr.Substring(0, groupNameStr.Length - 1);
  1668. }
  1669. }
  1670. item.DelegationStr = groupNameStr;
  1671. }
  1672. //获取模板
  1673. string tempPath = (AppSettingsHelper.Get("ExcelBasePath") + "Template/商邀资料模板.xls");
  1674. var designer = new WorkbookDesigner();
  1675. designer.Workbook = new Workbook(tempPath);
  1676. var dt = CommonFun.ToDataTableArray(_ivitiesViews);
  1677. dt.TableName = $"OADataView";
  1678. designer.SetDataSource(dt);
  1679. designer.Process();
  1680. //文件名
  1681. string fileName = $"商邀资料{DateTime.Now.ToString("yyyyMMddHHmmss")}.xls";
  1682. designer.Workbook.Save(AppSettingsHelper.Get("ExcelBasePath") + "InvitationOfficialActivityExport/" + fileName);
  1683. string url = AppSettingsHelper.Get("ExcelBaseUrl") + "Office/Excel/InvitationOfficialActivityExport/" + fileName;
  1684. return Ok(JsonView(true, "成功", url));
  1685. }
  1686. catch (Exception ex)
  1687. {
  1688. return Ok(JsonView(false, ex.Message));
  1689. }
  1690. }
  1691. /// <summary>
  1692. /// 根据商邀资料Id查询信息
  1693. /// </summary>
  1694. /// <param name="id"></param>
  1695. /// <returns></returns>
  1696. [HttpGet("{id}")]
  1697. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  1698. public async Task<IActionResult> QueryInvitationOfficialActivityById(int id)
  1699. {
  1700. if (id < 1) return Ok(JsonView(false, "Id参数错误!"));
  1701. return Ok(await _InvitationOfficialActivityDataRep.Info(id));
  1702. }
  1703. /// <summary>
  1704. /// 商邀资料操作(Status:1.新增,2.修改)
  1705. /// </summary>
  1706. /// <param name="dto"></param>
  1707. /// <returns></returns>
  1708. [HttpPost]
  1709. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  1710. public async Task<IActionResult> OpInvitationOfficialActivity([FromForm] OpInvitationOfficialActivityDto dto)
  1711. {
  1712. return Ok(await _InvitationOfficialActivityDataRep.IOA_OP(dto));
  1713. }
  1714. /// <summary>
  1715. /// 商邀资料 删除文件
  1716. /// </summary>
  1717. /// <param name="id"></param>
  1718. /// <param name="fileName"></param>
  1719. /// <returns></returns>
  1720. [HttpDelete("{id}")]
  1721. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  1722. public async Task<IActionResult> InvitationOfficialActivityDelFile(int id, string fileName)
  1723. {
  1724. if (id < 1) return Ok(JsonView(false, "请传入有效的Id"));
  1725. if (string.IsNullOrEmpty(fileName)) return Ok(JsonView(false, "文件名称不能为空!"));
  1726. var info = await _sqlSugar.Queryable<Res_InvitationOfficialActivityData>().FirstAsync(x => x.Id == id);
  1727. if (info == null) return Ok(JsonView(false, "该条数据不存在!"));
  1728. var files = new List<string>();
  1729. try
  1730. {
  1731. files = JsonConvert.DeserializeObject<List<string>>(info.SndFileName);
  1732. }
  1733. catch (Exception) { }
  1734. if (files != null && files.Count > 0 && files.Contains(fileName))
  1735. {
  1736. var filePath = $@"{AppSettingsHelper.Get("GrpFileBasePath")}/商邀相关文件/{fileName}";
  1737. if (System.IO.File.Exists(filePath))
  1738. {
  1739. System.IO.File.Delete(filePath);
  1740. }
  1741. //更改文件值
  1742. files.Remove(fileName);
  1743. var fileUpd = await _sqlSugar.Updateable<Res_InvitationOfficialActivityData>()
  1744. .SetColumns(x => x.SndFileName == JsonConvert.SerializeObject(files))
  1745. .Where(x => x.Id == id)
  1746. .ExecuteCommandAsync();
  1747. if (fileUpd > 0) return Ok(JsonView(true, "操作成功!"));
  1748. }
  1749. return Ok(JsonView(false, "操作失败!"));
  1750. }
  1751. /// <summary>
  1752. /// 删除商邀资料信息
  1753. /// </summary>
  1754. /// <param name="dto"></param>
  1755. /// <returns></returns>
  1756. [HttpPost]
  1757. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  1758. public async Task<IActionResult> DelInvitationOfficialActivity(DelBaseDto dto)
  1759. {
  1760. var res = await _InvitationOfficialActivityDataRep.SoftDeleteByIdAsync<Res_InvitationOfficialActivityData>(dto.Id.ToString(), dto.DeleteUserId);
  1761. if (!res) return Ok(JsonView(false, "删除失败"));
  1762. var info = await _sqlSugar.Queryable<Res_InvitationOfficialActivityData>().FirstAsync(x => x.Id == dto.Id);
  1763. if (info == null) return Ok(JsonView(false, "该条数据不存在!"));
  1764. var files = new List<string>();
  1765. try
  1766. {
  1767. files = JsonConvert.DeserializeObject<List<string>>(info.SndFileName);
  1768. }
  1769. catch (Exception) { }
  1770. if (files != null && files.Count > 0)
  1771. {
  1772. foreach (var fileName in files)
  1773. {
  1774. var filePath = $@"{AppSettingsHelper.Get("GrpFileBasePath")}/商邀相关文件/{fileName}";
  1775. if (System.IO.File.Exists(filePath))
  1776. {
  1777. System.IO.File.Delete(filePath);
  1778. }
  1779. }
  1780. }
  1781. return Ok(JsonView(true, "删除成功!"));
  1782. }
  1783. /// <summary>
  1784. /// 商邀资料 ExcelExport
  1785. /// </summary>
  1786. /// <param name="dto"></param>
  1787. /// <returns></returns>
  1788. [HttpPost]
  1789. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  1790. public async Task<IActionResult> InvitationOfficialActivityExcelExport(QueryInvitationOfficialActivityDataDto dto)
  1791. {
  1792. var ExcludedKeyStr = new string[] { "快递费" };
  1793. var ClientKeyStr = new string[] { "邀请函翻译" };
  1794. var columns = "Id,UnitName,Contact";
  1795. #region 参数验证
  1796. if (dto.PageIndex < 1) return Ok(JsonView(false, "请传入有效的PageIndex参数!"));
  1797. if (dto.PageSize < 1) return Ok(JsonView(false, "请传入有效的PageSize参数!"));
  1798. #endregion
  1799. if (!string.IsNullOrWhiteSpace(dto.Country)) { columns += ",Country"; }
  1800. if (!string.IsNullOrWhiteSpace(dto.Delegation)) { columns += ",Delegation"; }
  1801. if (!string.IsNullOrWhiteSpace(dto.Field)) { columns += ",Field"; }
  1802. DateTime endTime = new DateTime();
  1803. var InvitationOfficialActivityDataList = _sqlSugar.Queryable<Res_InvitationOfficialActivityData>()
  1804. .Where(x => x.IsDel == 0)
  1805. .WhereIF(dto.CreateUserId != 0 && !string.IsNullOrWhiteSpace(dto.CreateUserId.ToString()), x => x.CreateUserId == dto.CreateUserId)
  1806. .WhereIF(DateTime.TryParse(dto.StartCreateTime, out DateTime startTime) && DateTime.TryParse(dto.EndCreateTime, out endTime), x => x.CreateTime >= startTime && x.CreateTime <= endTime)
  1807. .Select(columns)
  1808. .ToList();
  1809. foreach (var item in InvitationOfficialActivityDataList)
  1810. {
  1811. EncryptionProcessor.DecryptProperties(item);
  1812. }
  1813. InvitationOfficialActivityDataList = InvitationOfficialActivityDataList
  1814. .WhereIF(!string.IsNullOrWhiteSpace(dto.Country), x => !string.IsNullOrWhiteSpace(x.Country) && x.Country.Contains(dto.Country))
  1815. .WhereIF(!string.IsNullOrWhiteSpace(dto.UnitName), x => !string.IsNullOrWhiteSpace(x.UnitName) && x.UnitName.Contains(dto.UnitName))
  1816. .WhereIF(!string.IsNullOrWhiteSpace(dto.Contact), x => !string.IsNullOrWhiteSpace(x.Contact) && x.Contact.Contains(dto.Contact))
  1817. .WhereIF(!string.IsNullOrWhiteSpace(dto.Delegation), x => !string.IsNullOrWhiteSpace(x.Delegation) && x.Delegation.Contains(dto.Delegation))
  1818. .WhereIF(!string.IsNullOrWhiteSpace(dto.Field), x => !string.IsNullOrWhiteSpace(x.Field) && x.Field.Contains(dto.Field))
  1819. .ToList();
  1820. foreach (var item in ExcludedKeyStr)
  1821. {
  1822. InvitationOfficialActivityDataList = InvitationOfficialActivityDataList.Where(x => string.IsNullOrWhiteSpace(x.UnitName) || !x.UnitName.Contains(item)).ToList();
  1823. //sqlWhere += $" And i.UnitName not like '%{item}%' ";
  1824. }
  1825. foreach (var item in ClientKeyStr)
  1826. {
  1827. InvitationOfficialActivityDataList = InvitationOfficialActivityDataList.Where(x => string.IsNullOrWhiteSpace(x.Contact) || !x.Contact.Contains(item)).ToList();
  1828. //sqlWhere += $" And i.Contact not like '%{item}%' ";
  1829. }
  1830. var ids = InvitationOfficialActivityDataList.Select(x => x.Id);
  1831. var _ivitiesViews = await _sqlSugar.Queryable<Res_InvitationOfficialActivityData>()
  1832. .LeftJoin<Sys_Users>((a, b) => b.IsDel == 0 && a.CreateUserId == b.Id)
  1833. .Where((a, b) => ids.Contains(a.Id))
  1834. .OrderByDescending((a, b) => a.CreateTime)
  1835. .Select((a, b) => new IOAInfoView
  1836. {
  1837. FaceBook = a.FaceBook,
  1838. Id = a.Id,
  1839. Address = a.Address,
  1840. Background = a.Background,
  1841. City = a.City,
  1842. Contact = a.Contact,
  1843. Country = a.Country,
  1844. Delegation = a.Delegation,
  1845. Email = a.Email,
  1846. Fax = a.Fax,
  1847. Field = a.Field,
  1848. Ins = a.Ins,
  1849. WeChat = a.WeChat,
  1850. UnitWeb = a.UnitWeb,
  1851. UnitName = a.UnitName,
  1852. UnitInfo = a.UnitInfo,
  1853. Tel = a.Tel,
  1854. SndFileName = a.SndFileName,
  1855. Remark = a.Remark,
  1856. OtherInfo = a.OtherInfo,
  1857. Job = a.Job
  1858. })
  1859. .ToListAsync();
  1860. if (!_ivitiesViews.Any()) return Ok(JsonView(false, $"您查询的数据暂无内容,不可导出!"));
  1861. var allGroupIds = new HashSet<int>();
  1862. foreach (var item in _ivitiesViews)
  1863. {
  1864. EncryptionProcessor.DecryptProperties(item);
  1865. if (!string.IsNullOrEmpty(item.Delegation))
  1866. {
  1867. allGroupIds.UnionWith(
  1868. item.Delegation.Split(',')
  1869. .Select(x =>
  1870. {
  1871. int id;
  1872. if (int.TryParse(x, out id)) return id;
  1873. return 0;
  1874. })
  1875. .Where(id => id != 0)
  1876. );
  1877. }
  1878. }
  1879. var _DelegationInfos = _sqlSugar.Queryable<Grp_DelegationInfo>()
  1880. .Where(x => allGroupIds.Contains(x.Id) && x.IsDel == 0)
  1881. .ToList()
  1882. .GroupBy(x => x.Id)
  1883. .ToDictionary(group => group.Key, group => group.Select(g => g.TeamName));
  1884. foreach (var item in _ivitiesViews)
  1885. {
  1886. string groupNameStr = "";
  1887. if (!string.IsNullOrEmpty(item.Delegation))
  1888. {
  1889. var groupIds = item.Delegation.Split(',').Select(x =>
  1890. {
  1891. int id;
  1892. if (int.TryParse(x, out id)) return id;
  1893. return 0;
  1894. })
  1895. .Where(id => id != 0)
  1896. .ToArray();
  1897. foreach (var id in groupIds)
  1898. {
  1899. if (_DelegationInfos.ContainsKey(id))
  1900. {
  1901. groupNameStr += string.Join("", _DelegationInfos[id]) + ",";
  1902. }
  1903. }
  1904. if (groupNameStr.Length > 1)
  1905. {
  1906. groupNameStr = groupNameStr.Substring(0, groupNameStr.Length - 1);
  1907. }
  1908. }
  1909. item.DelegationStr = groupNameStr;
  1910. }
  1911. //获取模板
  1912. string tempPath = (AppSettingsHelper.Get("ExcelBasePath") + "Template/商邀资料模板.xls");
  1913. var designer = new WorkbookDesigner();
  1914. designer.Workbook = new Workbook(tempPath);
  1915. var dt = CommonFun.ToDataTableArray(_ivitiesViews);
  1916. dt.TableName = $"OADataView";
  1917. designer.SetDataSource(dt);
  1918. designer.Process();
  1919. //文件名
  1920. string fileName = $"商邀资料{DateTime.Now.ToString("yyyyMMddHHmmss")}.xls";
  1921. designer.Workbook.Save(AppSettingsHelper.Get("ExcelBasePath") + "InvitationOfficialActivityExport/" + fileName);
  1922. string url = AppSettingsHelper.Get("ExcelBaseUrl") + "Office/Excel/InvitationOfficialActivityExport/" + fileName;
  1923. return Ok(JsonView(true, "成功", url));
  1924. }
  1925. /// <summary>
  1926. /// 商邀资料 导入文件(Excel)模板下载
  1927. /// </summary>
  1928. /// <returns></returns>
  1929. [HttpGet]
  1930. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  1931. public async Task<IActionResult> InvitationOfficialActivityExcelTemplateExport()
  1932. {
  1933. //获取模板
  1934. string fileName = $"商邀资料上传文件模板.xls";
  1935. string url = AppSettingsHelper.Get("ExcelBaseUrl") + "Office/Excel/Template/" + fileName;
  1936. return Ok(JsonView(true, "成功", url));
  1937. }
  1938. /// <summary>
  1939. /// 商邀资料 上传Excel文件添加
  1940. /// </summary>
  1941. /// <param name="file"></param>
  1942. /// <param name="currUserId"></param>
  1943. /// <returns></returns>
  1944. [HttpPost]
  1945. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  1946. public async Task<IActionResult> InvitationOfficialActivityExcelUpload(IFormFile file, int currUserId)
  1947. {
  1948. var fileName = file.FileName;
  1949. var filePostfix = fileName.Split(".")[1].ToUpper();
  1950. if (!filePostfix.Contains("XLS")) return Ok(JsonView(false, $"请传入Excel文件"));
  1951. if (file.Length < 1) return Ok(JsonView(false, $"请传入有效的文件"));
  1952. if (currUserId < 1) return Ok(JsonView(false, $"请传入有效的UserId"));
  1953. //保存文件
  1954. string fileDirPath = string.Format(@"{0}InvitationOfficialActivityExcelUpload/", AppSettingsHelper.Get("ExcelBasePath"));
  1955. if (!Directory.Exists(fileDirPath))
  1956. {
  1957. Directory.CreateDirectory(fileDirPath);
  1958. }
  1959. var filePath = Path.Combine(fileDirPath, fileName);
  1960. await using var stream = new FileStream(filePath, FileMode.Create);
  1961. await file.CopyToAsync(stream);
  1962. //读取保存的文件
  1963. Workbook workbook = new Workbook(filePath);
  1964. if (workbook == null) return Ok(JsonView(false, $"Excel文件不存在!"));
  1965. Worksheet worksheet = workbook.Worksheets[0];
  1966. if (worksheet == null) return Ok(JsonView(false, $"Excel文件工作簿不存在!"));
  1967. var infos = new List<Res_InvitationOfficialActivityData>();
  1968. var groupNames = await _sqlSugar.Queryable<Grp_DelegationInfo>().Where(x => x.IsDel == 0 && !string.IsNullOrEmpty(x.TeamName)).Select(x => new { x.Id, x.TeamName }).ToListAsync();
  1969. for (int row = 1; row < worksheet.Cells.MaxDataRow + 1; row++)
  1970. {
  1971. var groupIdsLabel = string.Empty;
  1972. var groupNames1 = worksheet.Cells[row, 14].StringValue ?? "-";
  1973. if (!string.IsNullOrEmpty(groupNames1))
  1974. {
  1975. if (groupNames1.Contains(";"))
  1976. {
  1977. var groupNameArray = groupNames1.Split(";").Where(x => !string.IsNullOrEmpty(x)).ToList();
  1978. var groupIds = groupNames.Where(x => groupNameArray.Contains(x.TeamName)).Select(x => x.Id).ToList();
  1979. if (groupIds.Any()) groupIdsLabel = string.Join(',', groupIds);
  1980. }
  1981. else groupIdsLabel = groupNames.Where(x => groupNames1.Equals(x.TeamName)).FirstOrDefault()?.Id.ToString() ?? string.Empty;
  1982. }
  1983. infos.Add(new Res_InvitationOfficialActivityData()
  1984. {
  1985. Country = worksheet.Cells[row, 0].StringValue ?? "-",
  1986. City = worksheet.Cells[row, 1].StringValue ?? "-",
  1987. UnitName = worksheet.Cells[row, 2].StringValue ?? "-",
  1988. Field = worksheet.Cells[row, 3].StringValue ?? "-",
  1989. Address = worksheet.Cells[row, 4].StringValue ?? "-",
  1990. UnitInfo = worksheet.Cells[row, 5].StringValue ?? "-",
  1991. Contact = worksheet.Cells[row, 6].StringValue ?? "-",
  1992. Job = worksheet.Cells[row, 7].StringValue ?? "-",
  1993. Tel = worksheet.Cells[row, 8].StringValue ?? "-",
  1994. Email = worksheet.Cells[row, 9].StringValue ?? "-",
  1995. WeChat = worksheet.Cells[row, 10].StringValue ?? "-",
  1996. FaceBook = worksheet.Cells[row, 11].StringValue ?? "-",
  1997. Ins = worksheet.Cells[row, 12].StringValue ?? "-",
  1998. Fax = worksheet.Cells[row, 13].StringValue ?? "-",
  1999. Delegation = groupIdsLabel,
  2000. SndFilePath = worksheet.Cells[row, 15].StringValue ?? "-",
  2001. OtherInfo = worksheet.Cells[row, 16].StringValue ?? "-",
  2002. Remark = $"[{DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss")}] Excel上传文件添加! 文件存储地址[{filePath}]",
  2003. CreateUserId = currUserId,
  2004. IsDel = 0
  2005. });
  2006. }
  2007. //加密
  2008. foreach (var item in infos) EncryptionProcessor.EncryptProperties(item);
  2009. var addCount = await _sqlSugar.Insertable(infos).ExecuteCommandAsync();
  2010. if (addCount < 1) return Ok(JsonView(false, "文件上传添加失败!"));
  2011. return Ok(JsonView(true, $"上传文件添加成功!共添加{addCount}条!"));
  2012. }
  2013. #endregion
  2014. #region 商邀资料 AI
  2015. /// <summary>
  2016. /// 商邀资料AI 基础数据源
  2017. /// </summary>
  2018. /// <returns></returns>
  2019. [HttpGet]
  2020. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  2021. public async Task<IActionResult> InvitationAIInit()
  2022. {
  2023. var itemNames = await InvitationAIInvName();
  2024. var unitNames = await InvitationAIClientName();
  2025. var countries = await InvitationAICountryName();
  2026. var industryNodes = IndustryTree.Build().Select(x => x.NameCn).ToList();
  2027. var orgScales = OrgScale.BuildInitialData().Select(x => x.Name).ToList();
  2028. var orgLevels = new List<string>() {
  2029. "全部",
  2030. "总部",
  2031. "分公司"
  2032. };
  2033. return Ok(JsonView(true, $"查询成功!", new
  2034. {
  2035. itemNames,
  2036. unitNames,
  2037. countries,
  2038. industryNodes,
  2039. orgScales,
  2040. orgLevels
  2041. }));
  2042. }
  2043. /// <summary>
  2044. /// 商邀资料AI 资料列表
  2045. /// </summary>
  2046. /// <returns></returns>
  2047. [HttpGet("{name}")]
  2048. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  2049. public async Task<IActionResult> InvitationAIItemByName(string name)
  2050. {
  2051. if (string.IsNullOrWhiteSpace(name)) return Ok(JsonView(false, "名称无效"));
  2052. // 1. 获取主表记录
  2053. var info = await _sqlSugar.Queryable<Res_InvitationAI>()
  2054. .FirstAsync(x => x.IsDel == 0 && x.InvName == name);
  2055. var groupInfo = await _sqlSugar.Queryable<Grp_DelegationInfo>().FirstAsync(x => x.IsDel == 0 && x.TeamName.Equals(name));
  2056. string baseUrl = AppSettingsHelper.Get("OfficeBaseUrl")?.TrimEnd('/');
  2057. // 2. 分支处理:若无主表记录,尝试从团组表同步基本信息
  2058. if (info == null)
  2059. {
  2060. return Ok(JsonView(true, "暂无数据", new
  2061. {
  2062. Id = 0,
  2063. GroupId = groupInfo?.Id ?? 0,
  2064. InvName = name,
  2065. AiCrawledDetails = new List<InvitationAIInfo>(),
  2066. Entry = new
  2067. {
  2068. Objective = groupInfo?.VisitPurpose ?? "商务会谈",
  2069. OriginUnit = groupInfo?.ClientUnit ?? "",
  2070. TargetCountry = _delegationInfoRep.GroupSplitCountry(groupInfo?.VisitCountry ?? "")
  2071. }
  2072. }));
  2073. }
  2074. // 3. 数据填充(使用 ??= 语法确保 Null 安全)
  2075. info.EntryInfo ??= new EntryInfo();
  2076. if (string.IsNullOrEmpty(info.EntryInfo.Objective))
  2077. info.EntryInfo.Objective = groupInfo?.VisitPurpose ?? "商务会谈";
  2078. if (string.IsNullOrEmpty(info.EntryInfo.OriginUnit))
  2079. info.EntryInfo.OriginUnit = groupInfo?.ClientUnit ?? "";
  2080. if (info.EntryInfo.TargetCountry?.Any() != true)
  2081. info.EntryInfo.TargetCountry = _delegationInfoRep.GroupSplitCountry(groupInfo?.VisitCountry ?? "");
  2082. // 4. 排序:ThenByDescending 确保多级排序生效
  2083. info.AiCrawledDetails = info.AiCrawledDetails
  2084. .OrderByDescending(x => x.IsChecked)
  2085. .ThenByDescending(x => x.OperatedAt)
  2086. .ToList();
  2087. var details = info?.AiCrawledDetails ?? new List<InvitationAIInfo>();
  2088. if (details.Any())
  2089. {
  2090. foreach (var detail in details)
  2091. {
  2092. // 附件路径处理
  2093. if (detail.EmailInfo?.AttachmentPaths?.Any() == true)
  2094. {
  2095. detail.EmailInfo.AttachmentPaths = detail.EmailInfo.AttachmentPaths
  2096. .Select(p => p.StartsWith("http") ? p : $"{baseUrl}/{p.TrimStart('/')}")
  2097. .ToList();
  2098. }
  2099. // 官网地址注入新闻首位 (确保 PostUrl 已实例化)
  2100. if (!string.IsNullOrEmpty(detail.SiteUrl))
  2101. {
  2102. detail.PostUrl ??= new List<PostNewsItem>();
  2103. // 避免重复插入
  2104. if (!detail.PostUrl.Any(x => x.Description == "官网"))
  2105. {
  2106. detail.PostUrl.Insert(0, new PostNewsItem
  2107. {
  2108. Date = "-",
  2109. Description = "官网",
  2110. Url = detail.SiteUrl
  2111. });
  2112. }
  2113. }
  2114. }
  2115. }
  2116. return Ok(JsonView(true, "查询成功!", new
  2117. {
  2118. info.Id,
  2119. info.GroupId,
  2120. info.InvName,
  2121. AiCrawledDetails = details,
  2122. Entry = info.EntryInfo
  2123. }));
  2124. }
  2125. /// <summary>
  2126. /// 商邀资料AI 设置词条
  2127. /// </summary>
  2128. /// <returns></returns>
  2129. [HttpPost]
  2130. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  2131. public async Task<IActionResult> InvitationAISetPrompt(InvitationAISetPromptDto dto)
  2132. {
  2133. // 基础校验
  2134. if (string.IsNullOrWhiteSpace(dto.InvName) || string.IsNullOrWhiteSpace(dto.Objective) || string.IsNullOrWhiteSpace(dto.OriginUnit) ||
  2135. dto.TargetCountry == null || dto.TargetCountry.Count == 0 || dto.Industries == null || dto.Industries.Count == 0 ||
  2136. dto.ScaleTypes == null || dto.ScaleTypes.Count == 0
  2137. )
  2138. return Ok(JsonView(false, "请传入有效的公务名称、单位名称、国家、出访目的、行业和规模类型!"));
  2139. var invName = dto.InvName;
  2140. var groupInfo = await _sqlSugar.Queryable<Grp_DelegationInfo>()
  2141. .Where(x => x.IsDel == 0 && x.TeamName.Equals(invName))
  2142. .Select(x => new {x.Id,x.TeamName, x.VisitPurpose })
  2143. .FirstAsync();
  2144. int groupId = groupInfo?.Id ?? 0;
  2145. #region 数据库操作
  2146. // 数据库信息获取方式
  2147. var dataInfo = await _sqlSugar.Queryable<Res_InvitationAI>().Where(x => x.IsDel == 0 && x.InvName.Equals( invName)).FirstAsync();
  2148. #region 词条信息
  2149. string operatorName = await _sqlSugar.Queryable<Sys_Users>().Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  2150. var entryInfo = new EntryInfo()
  2151. {
  2152. OriginUnit = dto.OriginUnit,
  2153. TargetCountry = dto.TargetCountry,
  2154. Objective = dto.Objective,
  2155. Industries = dto.Industries,
  2156. ScaleTypes = dto.ScaleTypes,
  2157. IsBackground = dto.IsBackground,
  2158. OrgLevel = dto.OrgLevel,
  2159. OtherConstraints = dto.OtherConstraints,
  2160. Operator = operatorName,
  2161. OperatedAt = DateTime.Now
  2162. };
  2163. #endregion
  2164. // 3.如果以上两种方式都没有查询到数据,则说明是新数据,需要添加到数据库
  2165. if (dataInfo == null)
  2166. {
  2167. // 3.1 新数据,需要添加到数据库
  2168. dataInfo = new Res_InvitationAI()
  2169. {
  2170. InvName = invName,
  2171. GroupId = groupId,
  2172. EntryInfo = entryInfo,
  2173. CreateUserId = dto.CurrUserId
  2174. };
  2175. var insert = await _sqlSugar.Insertable(dataInfo).ExecuteReturnIdentityAsync();
  2176. if (insert < 1)
  2177. {
  2178. return Ok(JsonView(false, $"词条信息新增失败!"));
  2179. }
  2180. dataInfo.Id = insert;
  2181. }
  2182. else
  2183. {
  2184. // 3.2 数据存在 则更新数据(覆盖原有的AI爬取详情,保留原有的其他字段不变)
  2185. dataInfo.EntryInfo = entryInfo;
  2186. var update = await _sqlSugar.Updateable(dataInfo).ExecuteCommandAsync();
  2187. if (update < 1)
  2188. {
  2189. return Ok(JsonView(false, $"词条信息更新失败!"));
  2190. }
  2191. }
  2192. #endregion
  2193. return Ok(JsonView(true, $"设置成功!", new
  2194. {
  2195. dataInfo.Id,
  2196. dataInfo.InvName,
  2197. dataInfo.GroupId,
  2198. Entry = dataInfo.EntryInfo,
  2199. }));
  2200. }
  2201. /// <summary>
  2202. /// 商邀资料AI 设置复选框选中 批量
  2203. /// </summary>
  2204. /// <param name="dto"></param>
  2205. /// <returns></returns>
  2206. [HttpPost]
  2207. public async Task<IActionResult> InvitationAISetChecked([FromBody] InvitationAISetCheckedDto dto)
  2208. {
  2209. // 1. 基础参数校验 (注意:这里移除了对 Guids.Any() 的强校验,允许空集合)
  2210. if (dto.Id < 1 || dto.CurrUserId < 1 || dto.Guids == null)
  2211. return Ok(JsonView(false, "请求参数不完整"));
  2212. // 2. 获取主记录
  2213. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI>()
  2214. .FirstAsync(x => x.IsDel == 0 && x.Id == dto.Id);
  2215. if (invAiInfo?.AiCrawledDetails == null || !invAiInfo.AiCrawledDetails.Any())
  2216. return Ok(JsonView(false, "数据不存在或集合为空"));
  2217. // 3. 准备更新所需数据
  2218. var guidSet = dto.Guids.ToHashSet();
  2219. var now = DateTime.Now;
  2220. // 获取操作人姓名
  2221. string operatorName = await _sqlSugar.Queryable<Sys_Users>()
  2222. .Where(x => x.Id == dto.CurrUserId)
  2223. .Select(x => x.CnName)
  2224. .FirstAsync() ?? "-";
  2225. // 4. 单次遍历完成 重置 + 更新
  2226. // 如果 guidSet 为空,循环会将所有项的 IsChecked 置为 false
  2227. foreach (var item in invAiInfo.AiCrawledDetails)
  2228. {
  2229. if (guidSet.Contains(item.Guid))
  2230. {
  2231. item.IsChecked = true;
  2232. }
  2233. else
  2234. {
  2235. item.IsChecked = false;
  2236. }
  2237. item.OperatedAt = now;
  2238. item.Operator = operatorName;
  2239. }
  2240. // 5. 排序逻辑
  2241. // 即使是全量取消选中,也可以按最后操作时间排序,让最近变动的项在前
  2242. invAiInfo.AiCrawledDetails = invAiInfo.AiCrawledDetails
  2243. .OrderByDescending(x => x.OperatedAt)
  2244. .ToList();
  2245. // 6. 提交数据库
  2246. // 注意:SqlSugar 的 UpdateColumns 会序列化整个对象列表并覆盖数据库字段
  2247. await _sqlSugar.Updateable(invAiInfo)
  2248. .UpdateColumns(x => x.AiCrawledDetails)
  2249. .ExecuteCommandAsync();
  2250. return Ok(JsonView(true, guidSet.Any() ? "设置成功" : "已全部取消选中"));
  2251. }
  2252. /// <summary>
  2253. /// 商邀资料AI 保存
  2254. /// </summary>
  2255. /// <returns></returns>
  2256. [HttpPost]
  2257. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  2258. public async Task<IActionResult> InvitationAISave(InvitationAISaveDto dto)
  2259. {
  2260. if (dto.Id < 1) return Ok(JsonView(false, "请先设置AI关键字"));
  2261. if (dto.CurrUserId < 1) return Ok(JsonView(false, "请传入用户Id"));
  2262. if (dto.AiCrawledDetail == null) return Ok(JsonView(false, "请选择保存的邀请方详情"));
  2263. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI>()
  2264. .Where(x => x.Id == dto.Id)
  2265. .FirstAsync();
  2266. if (invAiInfo == null) return Ok(JsonView(false, "数据信息为空"));
  2267. var dataList = invAiInfo?.AiCrawledDetails;
  2268. if (dataList == null) return Ok(JsonView(false, "邀请方数据信息为空"));
  2269. var datas = dataList.Where(x => x.Guid != dto.AiCrawledDetail.Guid).ToList();
  2270. var editInfo = dto.AiCrawledDetail;
  2271. // 如果 Guid 为空,说明是新增数据,需要生成新的 Guid
  2272. if (string.IsNullOrEmpty(editInfo.Guid))
  2273. {
  2274. editInfo.Guid = Guid.NewGuid().ToString("N");
  2275. editInfo.Source = 2; // 标记为手动新增数据
  2276. editInfo.OperatedAt = DateTime.Now;
  2277. }
  2278. var opUserName = await _sqlSugar.Queryable<Sys_Users>().Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  2279. editInfo.Operator = opUserName;
  2280. editInfo.OperatedAt = DateTime.Now;
  2281. datas.Add(editInfo);
  2282. invAiInfo.AiCrawledDetails = datas.OrderByDescending(x => x.OperatedAt).ToList();
  2283. var editUpd = await _sqlSugar.Updateable(invAiInfo).UpdateColumns(x => x.AiCrawledDetails).ExecuteCommandAsync();
  2284. if (editUpd < 1)
  2285. {
  2286. return Ok(JsonView(true, "保存失败"));
  2287. }
  2288. return Ok(JsonView(true, "保存成功"));
  2289. }
  2290. /// <summary>
  2291. /// 商邀资料AI 单条删除
  2292. /// </summary>
  2293. /// <returns></returns>
  2294. [HttpPost]
  2295. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  2296. public async Task<IActionResult> InvitationAISingleDel(InvitationAISingleDelDto dto)
  2297. {
  2298. int id = dto.Id;
  2299. string guid = dto.Guid;
  2300. // 基础校验
  2301. if (id < 1 || string.IsNullOrWhiteSpace(guid))
  2302. return Ok(JsonView(false, "请传入有效的Id和Guid!"));
  2303. var dataInfo = await _sqlSugar.Queryable<Res_InvitationAI>().Where(x => x.IsDel == 0 && x.Id == id).FirstAsync();
  2304. if (dataInfo == null)
  2305. return Ok(JsonView(false, "当前数据信息不存在!"));
  2306. var newDataInfos = dataInfo.AiCrawledDetails.Where(x => x.Guid != guid).ToList();
  2307. dataInfo.AiCrawledDetails = newDataInfos.OrderByDescending(x => x.OperatedAt).ToList();
  2308. var update = await _sqlSugar.Updateable(dataInfo).ExecuteCommandAsync();
  2309. if (update < 1)
  2310. {
  2311. return Ok(JsonView(false, $"删除失败!"));
  2312. }
  2313. return Ok(JsonView(true, $"删除成功!", new
  2314. {
  2315. dataInfo.Id,
  2316. dataInfo.InvName,
  2317. dataInfo.GroupId,
  2318. dataInfo.AiCrawledDetails
  2319. }));
  2320. }
  2321. /// <summary>
  2322. /// 商邀资料AI AI查询出的数据添加至商邀资料库
  2323. /// </summary>
  2324. /// <returns></returns>
  2325. [HttpPost]
  2326. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  2327. public async Task<IActionResult> InvitationAIInsertResource([FromBody] InvitationAIInsertResourceDto dto)
  2328. {
  2329. // 基础校验
  2330. if (dto.Id < 1 || dto.CurrUserId < 1 || dto.Guids == null || !dto.Guids.Any())
  2331. return Ok(JsonView(false, "参数不完整,请选择有效的单位数据"));
  2332. // 获取主记录
  2333. var dataInfo = await _sqlSugar.Queryable<Res_InvitationAI>()
  2334. .FirstAsync(x => x.IsDel == 0 && x.Id == dto.Id);
  2335. if (dataInfo?.AiCrawledDetails == null)
  2336. return Ok(JsonView(false, "当前数据信息不存在!"));
  2337. // 筛选出待转正的 AI 数据 (使用 HashSet 优化匹配)
  2338. var guidSet = dto.Guids.ToHashSet();
  2339. var targetAiDetails = dataInfo.AiCrawledDetails
  2340. .Where(x => x.Source != 0 && guidSet.Contains(x.Guid))
  2341. .ToList();
  2342. if (!targetAiDetails.Any())
  2343. return Ok(JsonView(false, "未找到符合条件的 AI 获取信息,无法重复添加或来源错误!"));
  2344. // 准备入库数据(批量构建)
  2345. var now = DateTime.Now;
  2346. var opUserName = await _sqlSugar.Queryable<Sys_Users>().Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  2347. var rawLocalDatas = targetAiDetails.Select(item => {
  2348. // 修改内存中的状态为本地数据
  2349. item.Source = 0;
  2350. item.OperatedAt = now;
  2351. item.Operator = opUserName;
  2352. return new Res_InvitationOfficialActivityData()
  2353. {
  2354. // 批量加密操作
  2355. Country = AesEncryptionHelper.Encrypt(item.Region),
  2356. UnitName = AesEncryptionHelper.Encrypt(item.NameCn),
  2357. Address = AesEncryptionHelper.Encrypt(item.Address),
  2358. Field = AesEncryptionHelper.Encrypt(item.Scope),
  2359. Contact = AesEncryptionHelper.Encrypt(item.Contact),
  2360. Tel = AesEncryptionHelper.Encrypt(item.Phone),
  2361. Email = AesEncryptionHelper.Encrypt(item.Email),
  2362. UnitWeb = AesEncryptionHelper.Encrypt(item.SiteUrl),
  2363. LastUpdateUserId = dto.CurrUserId,
  2364. LastUpdateTime = now,
  2365. CreateUserId = dto.CurrUserId,
  2366. CreateTime = now
  2367. };
  2368. }).ToList();
  2369. // 开启事务执行双表更新
  2370. var result = await _sqlSugar.UseTranAsync(async () =>
  2371. {
  2372. // 插入本地资料库
  2373. await _sqlSugar.Insertable(rawLocalDatas).ExecuteCommandAsync();
  2374. // 更新 AI 主表的状态(全量覆盖 JSON 列)
  2375. await _sqlSugar.Updateable(dataInfo)
  2376. .UpdateColumns(x => x.AiCrawledDetails)
  2377. .ExecuteCommandAsync();
  2378. });
  2379. return Ok(result.IsSuccess
  2380. ? JsonView(true, "成功添加至资料库并更新来源状态")
  2381. : JsonView(false, $"操作失败:{result.ErrorMessage}"));
  2382. }
  2383. /// <summary>
  2384. /// 商邀资料AI 混元AI查询资料(SSE流式推送)
  2385. /// </summary>
  2386. [HttpPost]
  2387. public async Task InvitationAISearchStreamProgress([FromBody] InvitationAISearchDto dto)
  2388. {
  2389. HttpContext.InitializeSse();
  2390. try
  2391. {
  2392. await HttpContext.SendSseStepAsync(5, "正在加载配置...");
  2393. #region 1. 异步并行化验证 (Performance Boost)
  2394. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI>().Where(x => x.IsDel == 0 && x.Id == dto.Id).FirstAsync();
  2395. var operatorName = await _sqlSugar.Queryable<Sys_Users>()
  2396. .Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId)
  2397. .Select(x => x.CnName).FirstAsync();
  2398. if (invAiInfo?.EntryInfo == null)
  2399. {
  2400. await HttpContext.SendSseStepAsync(-1, "未找到有效的关键字配置。");
  2401. return;
  2402. }
  2403. var entryInfo = invAiInfo.EntryInfo;
  2404. var targetCountrySet = new HashSet<string>(entryInfo.TargetCountry);
  2405. #endregion
  2406. await HttpContext.SendSseStepAsync(20, "正在解析本地典籍 (并行解密)...");
  2407. #region 2. 内存计算优化
  2408. // 仅查询必要字段,减少 DataReader 压力
  2409. var rawLocalDatas = await _sqlSugar.Queryable<Res_InvitationOfficialActivityData>()
  2410. .Where(x => x.IsDel == 0 && !string.IsNullOrEmpty(x.UnitName))
  2411. .Select(x => new { x.Country, x.UnitName, x.Address, x.Field, x.Contact, x.Tel, x.Email })
  2412. .ToListAsync();
  2413. // PLINQ 核心炼金:利用多核并行解密
  2414. var allDecrypted = rawLocalDatas.AsParallel().Select(item => new InvitationAIInfo
  2415. {
  2416. Guid = Guid.NewGuid().ToString("N"),
  2417. Source = 0, // 标识来源:本地
  2418. Region = AesEncryptionHelper.Decrypt(item.Country),
  2419. NameCn = AesEncryptionHelper.Decrypt(item.UnitName),
  2420. Address = AesEncryptionHelper.Decrypt(item.Address),
  2421. Scope = AesEncryptionHelper.Decrypt(item.Field),
  2422. Contact = AesEncryptionHelper.Decrypt(item.Contact),
  2423. Phone = AesEncryptionHelper.Decrypt(item.Tel),
  2424. Email = AesEncryptionHelper.Decrypt(item.Email),
  2425. OperatedAt = DateTime.Now,
  2426. Operator = operatorName,
  2427. }).ToList();
  2428. // 筛选符合国家的本地数据
  2429. var matchedCountries = allDecrypted.Where(x => targetCountrySet.Contains(x.Region)).ToList();
  2430. //var matchedCountries = new List<InvitationAIInfo>();
  2431. #endregion
  2432. #region 3. 混元 AI 协同炼金 (双阶段)
  2433. // --- 阶段 A: 行业匹配分析 ---
  2434. if (matchedCountries.Any())
  2435. {
  2436. await HttpContext.SendSseStepAsync(40, $"AI 正在执行 {entryInfo.TargetCountry.Count} 国的行业契合度分析...");
  2437. string industryQuestion = BuildIndustryPrompt(entryInfo, matchedCountries);
  2438. string industryRaw = await _hunyuanService.ChatCompletionsHunyuan_t1_latestAsync(industryQuestion);
  2439. var industryMatches = CleanAndParseJson<List<IndustryMatchResult>>(industryRaw) ?? new();
  2440. var matchedNames = new HashSet<string>(industryMatches.Select(x => x.TargetUnitName));
  2441. // 重新过滤:仅保留 AI 认为匹配的本地数据
  2442. matchedCountries = matchedCountries.Where(x => !string.IsNullOrEmpty(x.NameCn) && matchedNames.Contains(x.NameCn)).Distinct().ToList();
  2443. }
  2444. // --- 阶段 B: 差额补全 (Gap Filling) ---
  2445. var localInvDatas = new List<InvitationAIInfo>();
  2446. var aiTasks = new List<CountryAIPormptInfo>();
  2447. foreach (var country in entryInfo.TargetCountry)
  2448. {
  2449. var countryData = matchedCountries.Where(x => x.Region == country).Take(entryInfo.NeedCount).ToList();
  2450. localInvDatas.AddRange(countryData);
  2451. int gap = entryInfo.NeedCount - countryData.Count;
  2452. if (gap > 0) aiTasks.Add(new() { Country = country, Count = gap });
  2453. }
  2454. var hunyuanAIInvDatas = new List<InvitationAIInfo>();
  2455. if (aiTasks.Any())
  2456. {
  2457. // 强制冷却(应对 QPS 限制)
  2458. // 混元免费版或低阶版本通常有 1s/1次 的频率限制
  2459. await Task.Delay(1000);
  2460. // 任务配置计算
  2461. var countryTasks = QuotaScheduler.GenerateTasks(aiTasks, entryInfo.Industries, entryInfo.ScaleTypes);
  2462. var countryTasksGroupBy = countryTasks.GroupBy(x => x.Region)
  2463. .Select(g => new
  2464. {
  2465. Region = g.Key,
  2466. Counrt = g.Count()
  2467. })
  2468. .ToList();
  2469. await HttpContext.SendSseStepAsync(60, $"AI 正在跨境检索 {string.Join(", ", countryTasksGroupBy.Select(x => $"{x.Region}({x.Counrt}条)"))} 单位资料...");
  2470. string searchQuestion = BuildHunyuanPrompt(aiTasks, countryTasks, entryInfo);
  2471. _logger.LogInformation(@"公务名称:{InvName}; 混元AI查询提示词:{searchQuestion}", invAiInfo.InvName, searchQuestion);
  2472. string searchRaw = await _hunyuanService.ChatCompletionsHunyuan_t1_latestAsync(searchQuestion);
  2473. var audit = ParseHunyuanResult<InvitationAIInfo>(searchRaw);
  2474. if (audit.IsSuccess && audit.Data != null)
  2475. {
  2476. hunyuanAIInvDatas = audit.Data.Select(x =>
  2477. {
  2478. x.Guid = Guid.NewGuid().ToString("N");
  2479. x.Source = 1;
  2480. x.Operator = operatorName;
  2481. x.OperatedAt = DateTime.Now;
  2482. return x;
  2483. }).ToList();
  2484. await HttpContext.SendSseStepAsync(80,
  2485. $"AI 已完成跨境检索,共获取 {hunyuanAIInvDatas.Count} 条单位资料");
  2486. }
  2487. else
  2488. {
  2489. _logger.LogWarning(
  2490. "公务名称:{InvName};AI 检索失败;Step={Step};Reason={Reason}",
  2491. invAiInfo.InvName,
  2492. audit.AuditStep,
  2493. audit.AuditReason
  2494. );
  2495. await HttpContext.SendSseStepAsync(80,
  2496. $"{GetFrontendAuditTitle(audit.AuditStep!)}:" +
  2497. $"{GetFrontendAuditReason(audit.AuditReason!)}");
  2498. hunyuanAIInvDatas = new();
  2499. }
  2500. }
  2501. #endregion
  2502. await HttpContext.SendSseStepAsync(90, "正在同步成果至持久化仓库...");
  2503. #region 4. 数据融合与更新
  2504. var finalResult = localInvDatas.Concat(hunyuanAIInvDatas).ToList();
  2505. // 仅更新指定列,性能更优
  2506. invAiInfo.AiCrawledDetails = finalResult.OrderByDescending(x => x.OperatedAt).ToList();
  2507. bool isOk = await _sqlSugar.Updateable(invAiInfo)
  2508. .UpdateColumns(x => x.AiCrawledDetails)
  2509. .ExecuteCommandHasChangeAsync();
  2510. if (!isOk) await HttpContext.SendSseStepAsync(-1, $"数据持久化失败。");
  2511. #endregion
  2512. // 5. 终焉推送
  2513. string finalMsg = "操作成功!资料已全部就绪。";
  2514. // 如果 AI 全部失败,但本地有数据,仍可成功
  2515. if (!finalResult.Any())
  2516. {
  2517. finalMsg = "本次未检索到符合条件的境外单位,建议调整国家或行业范围。";
  2518. }
  2519. await HttpContext.SendSseStepAsync(100, finalMsg, new
  2520. {
  2521. invAiInfo.Id,
  2522. AiCrawledDetails = finalResult.OrderByDescending(x => x.Source).ThenBy(x => x.Region).ToList()
  2523. });
  2524. }
  2525. catch (Exception ex)
  2526. {
  2527. _logger.LogError(ex, "SSE 管道熔断");
  2528. await HttpContext.SendSseStepAsync(-1, $"SSE 错误:{ex.InnerException.Message}({ex.Message})");
  2529. }
  2530. finally
  2531. {
  2532. await HttpContext.FinalizeSseAsync();
  2533. }
  2534. }
  2535. /// <summary>
  2536. /// 清洗并解析 AI 返回的 JSON 块
  2537. /// </summary>
  2538. private static T? CleanAndParseJson<T>(string rawResponse)
  2539. {
  2540. if (string.IsNullOrWhiteSpace(rawResponse)) return default;
  2541. string cleanJson = rawResponse.Trim();
  2542. // 自动剥离 Markdown 语法糖
  2543. if (cleanJson.Contains("```json"))
  2544. cleanJson = cleanJson.Split("```json")[1].Split("```")[0];
  2545. else if (cleanJson.Contains("```"))
  2546. cleanJson = cleanJson.Split("```")[1].Split("```")[0];
  2547. return JsonConvert.DeserializeObject<T>(cleanJson.Trim());
  2548. }
  2549. /// <summary>
  2550. /// 商邀资料AI 混元AI续写(SSE流式推送)
  2551. /// </summary>
  2552. /// <param name="dto"></param>
  2553. /// <returns></returns>
  2554. [HttpPost]
  2555. public async Task InvitationAICompleteTextStream([FromBody] InvitationAICompleteTextDto dto)
  2556. {
  2557. HttpContext.InitializeSse();
  2558. try
  2559. {
  2560. // 初始化检查 (Progress: 5%)
  2561. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI>().Where(x => x.Id == dto.Id).FirstAsync();
  2562. if (invAiInfo?.EntryInfo == null)
  2563. {
  2564. await HttpContext.SendSseStepAsync(-1, "请先设置关键字信息!");
  2565. return;
  2566. }
  2567. await HttpContext.SendSseStepAsync(10, "任务初始化完成,正在计算补齐缺口...");
  2568. // 任务拆解逻辑
  2569. var entryInfo = invAiInfo.EntryInfo;
  2570. var aiTasks = new List<CountryAIPormptInfo>();
  2571. int targetPerCountry = entryInfo.NeedCount;
  2572. foreach (var countryName in entryInfo.TargetCountry)
  2573. {
  2574. var countryDataCount = invAiInfo.AiCrawledDetails.Count(x => x.Region == countryName);
  2575. int aiNeedCount = targetPerCountry - countryDataCount;
  2576. if (aiNeedCount > 0) aiTasks.Add(new() { Country = countryName, Count = aiNeedCount });
  2577. }
  2578. if (!aiTasks.Any())
  2579. {
  2580. await HttpContext.SendSseStepAsync(100, "数据已满额,无需续写", invAiInfo.AiCrawledDetails);
  2581. return;
  2582. }
  2583. // 准备 AI Prompt (Progress: 20%)
  2584. var existingNames = new HashSet<string>(invAiInfo.AiCrawledDetails.Where(x => x.Source == 1).Select(x => x.NameCn));
  2585. string promptOther = $"请基于以下已存在的名称列表进行推荐,避免重复:{string.Join(", ", existingNames)}。{entryInfo.OtherConstraints}";
  2586. entryInfo.OtherConstraints = promptOther; // 将去重提示注入 entryInfo,确保 Prompt 构建时包含该信息
  2587. // 构建 Question
  2588. // 任务配置计算
  2589. var countryTasks = QuotaScheduler.GenerateTasks(aiTasks, entryInfo.Industries, entryInfo.ScaleTypes);
  2590. string question = BuildHunyuanPrompt(aiTasks, countryTasks, entryInfo);
  2591. await HttpContext.SendSseStepAsync(30, "AI 正在深度检索跨境商邀数据,请稍候...");
  2592. // ====== 调用 AI ======
  2593. await HttpContext.SendSseStepAsync(70, "AI 正在跨境检索单位资料...");
  2594. string searchRaw = await _hunyuanService.ChatCompletionsHunyuan_t1_latestAsync(question);
  2595. await HttpContext.SendSseStepAsync(80, "数据已捕获,正在进行格式校验与去重...");
  2596. // ====== 解析 AI ======
  2597. var operatorName = await _sqlSugar.Queryable<Sys_Users>()
  2598. .Where(x => x.Id == dto.CurrUserId)
  2599. .Select(x => x.CnName)
  2600. .FirstAsync() ?? "-";
  2601. var audit = ParseHunyuanResult<InvitationAIInfo>(searchRaw);
  2602. // AI 检索失败
  2603. if (!audit.IsSuccess)
  2604. {
  2605. _logger.LogWarning(
  2606. "公务名称:{InvName};AI 检索失败;Step={Step};Reason={Reason}",
  2607. invAiInfo.InvName,
  2608. audit.AuditStep,
  2609. audit.AuditReason
  2610. );
  2611. await HttpContext.SendSseStepAsync(-1,
  2612. $"{GetFrontendAuditTitle(audit.AuditStep!)}:" +
  2613. $"{GetFrontendAuditReason(audit.AuditReason!)}");
  2614. return;
  2615. }
  2616. // ====== AI 成功但无数据 ======
  2617. var hunyuanAIInvDatas = audit.Data!
  2618. .Select(x =>
  2619. {
  2620. x.Guid = Guid.NewGuid().ToString("N");
  2621. x.Source = 1;
  2622. x.Operator = operatorName;
  2623. x.OperatedAt = DateTime.Now;
  2624. return x;
  2625. })
  2626. .ToList();
  2627. if (!hunyuanAIInvDatas.Any())
  2628. {
  2629. await HttpContext.SendSseStepAsync(-1,"本次未检索到符合条件的境外单位,建议调整国家或行业范围。");
  2630. return;
  2631. }
  2632. // ====== 数据库写入 ======
  2633. invAiInfo.AiCrawledDetails ??= new();
  2634. invAiInfo.AiCrawledDetails.AddRange(hunyuanAIInvDatas);
  2635. invAiInfo.AiCrawledDetails =
  2636. invAiInfo.AiCrawledDetails.OrderByDescending(x => x.OperatedAt).ToList();
  2637. var updated = await _sqlSugar.Updateable(invAiInfo)
  2638. .UpdateColumns(x => x.AiCrawledDetails)
  2639. .ExecuteCommandAsync();
  2640. if (updated <= 0)
  2641. {
  2642. await HttpContext.SendSseStepAsync(-1, "数据库更新失败,请联系管理员");
  2643. return;
  2644. }
  2645. await HttpContext.SendSseStepAsync(100,
  2646. $"AI 续写成功!新增 {hunyuanAIInvDatas.Count} 条数据",
  2647. invAiInfo.AiCrawledDetails);
  2648. }
  2649. catch (Exception ex)
  2650. {
  2651. _logger.LogError(ex, "SSE 续写异常");
  2652. await HttpContext.SendSseStepAsync(-1, $"炼金炸炉:{ex.Message}");
  2653. }
  2654. finally
  2655. {
  2656. await HttpContext.FinalizeSseAsync();
  2657. }
  2658. }
  2659. /// <summary>
  2660. /// 商邀资料AI 文件生成(基于混元AI已爬取数据进行格式化输出,供用户下载使用)
  2661. /// </summary>
  2662. /// <returns></returns>
  2663. [HttpGet("{id}")]
  2664. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  2665. public async Task<IActionResult> InvitationAIFileGenerator(int id)
  2666. {
  2667. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI>()
  2668. .Where(x => x.Id == id)
  2669. .FirstAsync();
  2670. var dataList = invAiInfo?.AiCrawledDetails;
  2671. if (dataList == null)
  2672. return Ok(JsonView(false, "数据为空"));
  2673. Document doc = new Document();
  2674. DocumentBuilder builder = new DocumentBuilder(doc);
  2675. // A4页面设置
  2676. double usableWidth = 495;
  2677. foreach (Aspose.Words.Section sec in doc.Sections)
  2678. {
  2679. sec.PageSetup.PaperSize = Aspose.Words.PaperSize.A4;
  2680. sec.PageSetup.LeftMargin = 50;
  2681. sec.PageSetup.RightMargin = 50;
  2682. sec.PageSetup.TopMargin = 50;
  2683. sec.PageSetup.BottomMargin = 50;
  2684. }
  2685. // 卡片背景
  2686. Color cardBg = Color.FromArgb(252, 252, 252);
  2687. string fontName = "微软雅黑";
  2688. int count = 1;
  2689. foreach (var item in dataList)
  2690. {
  2691. // 标题
  2692. builder.ParagraphFormat.ClearFormatting();
  2693. builder.Font.ClearFormatting();
  2694. builder.Font.Name = fontName;
  2695. builder.Font.Size = 16;
  2696. builder.Font.Bold = true;
  2697. builder.Font.Color = Color.Black;
  2698. builder.Write($"{count++}. {item.NameCn}");
  2699. builder.InsertParagraph();
  2700. // 英文名
  2701. builder.Font.Size = 10;
  2702. builder.Font.Bold = false;
  2703. builder.Font.Color = Color.FromArgb(120, 120, 120);
  2704. builder.Write(item.NameEn ?? "");
  2705. builder.InsertParagraph();
  2706. builder.InsertParagraph();
  2707. // ====== 卡片表格 ======
  2708. Aspose.Words.Tables.Table table = builder.StartTable();
  2709. builder.CellFormat.Shading.BackgroundPatternColor = cardBg;
  2710. builder.CellFormat.VerticalAlignment = CellVerticalAlignment.Center;
  2711. builder.CellFormat.WrapText = true;
  2712. builder.CellFormat.LeftPadding = 12;
  2713. builder.CellFormat.RightPadding = 12;
  2714. builder.CellFormat.TopPadding = 8;
  2715. builder.CellFormat.BottomPadding = 8;
  2716. double labelW = 120;
  2717. AddSingleRow(builder, fontName, labelW, usableWidth - labelW, "详细地址", item.Address);
  2718. AddSingleRow(builder, fontName, labelW, usableWidth - labelW, "联系人", item.Contact);
  2719. AddSingleRow(builder, fontName, labelW, usableWidth - labelW, "联系电话", item.Phone);
  2720. AddSingleRow(builder, fontName, labelW, usableWidth - labelW, "电子邮箱", item.Email);
  2721. AddSingleLinkRow(builder, fontName, labelW, usableWidth - labelW, "官方网站", item.SiteUrl);
  2722. AddSingleLinkRow(builder, fontName, labelW, usableWidth - labelW, "最近三年动态", string.Join(", ", item.PostUrl.Select(p => p.Url)));
  2723. AddFullWidthRow(builder, fontName, labelW, usableWidth - labelW, "推荐等级", item.RecLevel);
  2724. AddFullWidthRow(builder, fontName, labelW, usableWidth - labelW, "经营范围", item.Scope);
  2725. AddFullWidthRow(builder, fontName, labelW, usableWidth - labelW, "对接建议", item.IntgAdvice);
  2726. // 表格样式
  2727. table.SetBorders(Aspose.Words.LineStyle.Single, 0.5, Color.FromArgb(230, 230, 230));
  2728. table.AutoFit(AutoFitBehavior.FixedColumnWidths);
  2729. table.AllowAutoFit = false;
  2730. table.PreferredWidth = PreferredWidth.FromPoints(usableWidth);
  2731. builder.EndTable();
  2732. builder.InsertParagraph();
  2733. builder.InsertParagraph();
  2734. }
  2735. // 保存逻辑
  2736. string fileName = $"{invAiInfo.InvName}_Professional.docx";
  2737. string filePath = Path.Combine(AppSettingsHelper.Get("InvitationAIAssistBasePath"), fileName);
  2738. doc.Save(filePath, Aspose.Words.SaveFormat.Docx);
  2739. return Ok(JsonView(true, "生成成功", new { Url = $"{AppSettingsHelper.Get("WordBaseUrl")}/{AppSettingsHelper.Get("InvitationAIAssistFtpPath")}/{fileName}" }));
  2740. }
  2741. #region 15.12 适配助手方法
  2742. private void AddSingleRow(DocumentBuilder builder, string font, double labelW, double contentW, string label, string value)
  2743. {
  2744. // 标签
  2745. builder.InsertCell();
  2746. builder.CellFormat.Width = labelW;
  2747. builder.ParagraphFormat.Alignment = ParagraphAlignment.Right;
  2748. builder.Font.Name = font;
  2749. builder.Font.Size = 9;
  2750. builder.Font.Bold = false;
  2751. builder.Font.Color = Color.FromArgb(120, 120, 120);
  2752. builder.Write(label);
  2753. // 内容
  2754. builder.InsertCell();
  2755. builder.CellFormat.Width = contentW;
  2756. builder.CellFormat.HorizontalMerge = CellMerge.First;
  2757. builder.ParagraphFormat.Alignment = ParagraphAlignment.Left;
  2758. builder.Font.Size = 10;
  2759. builder.Font.Bold = false;
  2760. builder.Font.Color = Color.FromArgb(60, 60, 60);
  2761. builder.Write(string.IsNullOrEmpty(value) ? "—" : value);
  2762. // 合并剩余列
  2763. builder.InsertCell();
  2764. builder.CellFormat.HorizontalMerge = CellMerge.Previous;
  2765. builder.InsertCell();
  2766. builder.CellFormat.HorizontalMerge = CellMerge.Previous;
  2767. builder.EndRow();
  2768. builder.CellFormat.HorizontalMerge = CellMerge.None;
  2769. }
  2770. private void AddSingleLinkRow(DocumentBuilder builder, string font, double labelW, double contentW, string label, string url)
  2771. {
  2772. builder.InsertCell();
  2773. builder.CellFormat.Width = labelW;
  2774. builder.ParagraphFormat.Alignment = ParagraphAlignment.Right;
  2775. builder.Font.Size = 9;
  2776. builder.Font.Bold = false;
  2777. builder.Font.Color = Color.FromArgb(120, 120, 120);
  2778. builder.Write(label);
  2779. builder.InsertCell();
  2780. builder.CellFormat.Width = contentW;
  2781. builder.CellFormat.HorizontalMerge = CellMerge.First;
  2782. builder.ParagraphFormat.Alignment = ParagraphAlignment.Left;
  2783. if (!string.IsNullOrEmpty(url) && url.StartsWith("http"))
  2784. {
  2785. builder.Font.Size = 10;
  2786. builder.Font.Color = Color.FromArgb(0, 102, 204);
  2787. builder.Font.Underline = Underline.None;
  2788. builder.InsertHyperlink(url, url, false);
  2789. }
  2790. else
  2791. {
  2792. builder.Font.Color = Color.FromArgb(60, 60, 60);
  2793. builder.Write("—");
  2794. }
  2795. builder.InsertCell();
  2796. builder.CellFormat.HorizontalMerge = CellMerge.Previous;
  2797. builder.InsertCell();
  2798. builder.CellFormat.HorizontalMerge = CellMerge.Previous;
  2799. builder.EndRow();
  2800. builder.CellFormat.HorizontalMerge = CellMerge.None;
  2801. }
  2802. private void AddFullWidthRow(DocumentBuilder builder, string font, double lw, double cw, string label, string value)
  2803. {
  2804. // 标签
  2805. builder.InsertCell();
  2806. builder.CellFormat.Width = lw;
  2807. builder.ParagraphFormat.Alignment = ParagraphAlignment.Right;
  2808. builder.Font.Size = 9;
  2809. builder.Font.Bold = false;
  2810. builder.Font.Color = Color.FromArgb(120, 120, 120);
  2811. builder.Write(label);
  2812. // 内容
  2813. builder.InsertCell();
  2814. builder.CellFormat.Width = cw;
  2815. builder.CellFormat.HorizontalMerge = CellMerge.First;
  2816. builder.ParagraphFormat.Alignment = ParagraphAlignment.Left;
  2817. builder.Font.Size = 10;
  2818. builder.Font.Bold = false;
  2819. builder.Font.Color = Color.FromArgb(60, 60, 60);
  2820. builder.Write(string.IsNullOrEmpty(value) ? "—" : value);
  2821. builder.InsertCell();
  2822. builder.CellFormat.HorizontalMerge = CellMerge.Previous;
  2823. builder.InsertCell();
  2824. builder.CellFormat.HorizontalMerge = CellMerge.Previous;
  2825. builder.EndRow();
  2826. builder.CellFormat.HorizontalMerge = CellMerge.None;
  2827. }
  2828. #endregion
  2829. /// <summary>
  2830. /// 商邀资料AI 生成邮件(SSE流式推送)
  2831. /// </summary>
  2832. /// <returns></returns>
  2833. [HttpPost]
  2834. public async Task InvitationAIGenerateEmailStream([FromBody] InvitationAIGenerateEmailDto dto)
  2835. {
  2836. HttpContext.InitializeSse();
  2837. try
  2838. {
  2839. // 1. 基础校验与资源加载 (Progress: 5%)
  2840. if (dto.Id < 1 || dto.Guids == null || !dto.Guids.Any())
  2841. {
  2842. await HttpContext.SendSseStepAsync(-1, "请求参数不完整,请选择单位");
  2843. return;
  2844. }
  2845. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI>().Where(x => x.Id == dto.Id).FirstAsync();
  2846. if (invAiInfo?.AiCrawledDetails == null)
  2847. {
  2848. await HttpContext.SendSseStepAsync(-1, "基础商邀资料不存在");
  2849. return;
  2850. }
  2851. await HttpContext.SendSseStepAsync(10, "AI 正在分析考察团组背景与访问意图...");
  2852. // 2. 准备 AI 上下文数据 (Progress: 15%)
  2853. var clientInfoSources = invAiInfo.AiCrawledDetails.Where(x => dto.Guids.Contains(x.Guid)).ToList();
  2854. var clientInfosForAI = clientInfoSources.Select(x => new AICreateEmailInfo()
  2855. {
  2856. Guid = x.Guid,
  2857. NameCn = x.NameCn,
  2858. Scope = x.Scope
  2859. }).ToList();
  2860. var groupInfo = await _sqlSugar.Queryable<Grp_DelegationInfo>()
  2861. .Where(x => x.IsDel == 0 && x.Id == invAiInfo.GroupId)
  2862. .Select(x => new { x.TeamName, x.VisitPurpose, x.VisitDate })
  2863. .FirstAsync();
  2864. string operatorName = await _sqlSugar.Queryable<Sys_Users>()
  2865. .Where(x => x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  2866. // 3. 构建 Prompt 并调用 AI (Progress: 25%)
  2867. await HttpContext.SendSseStepAsync(30, $"AI 正在为 {clientInfosForAI.Count} 家单位撰写定制化正式邮件...");
  2868. // 构建生成邮件的 Prompt
  2869. string pormpt = BuildHunyuanEmailPrompt(
  2870. invAiInfo.EntryInfo?.OriginUnit ?? "",
  2871. groupInfo?.VisitPurpose ?? "",
  2872. groupInfo?.VisitDate.ToString("yyyy-MM-dd") ?? "",
  2873. clientInfosForAI
  2874. );
  2875. // 调用 AI (此处为阻塞式等待 AI 结果,若混元支持 Stream 可进一步拆解)
  2876. string aiResponse = await _hunyuanService.ChatCompletionsHunyuan_t1_latestAsync(pormpt);
  2877. await HttpContext.SendSseStepAsync(80, "邮件初稿已生成,正在进行 HTML 格式校验与转义处理...");
  2878. // 4. 解析结果 (Progress: 85%)
  2879. var hunyuanAIEmailDatas = new List<AICreateEmailInfo>();
  2880. if (!string.IsNullOrWhiteSpace(aiResponse))
  2881. {
  2882. // 预处理:过滤 AI 可能返回的 Markdown 标记
  2883. string cleanJson = aiResponse.Trim();
  2884. if (cleanJson.StartsWith("```json"))
  2885. cleanJson = cleanJson.Substring(7, cleanJson.Length - 10).Trim();
  2886. else if (cleanJson.StartsWith("```"))
  2887. cleanJson = cleanJson.Substring(3, cleanJson.Length - 6).Trim();
  2888. try
  2889. {
  2890. // 解析并注入 Source 标识
  2891. hunyuanAIEmailDatas = JsonConvert.DeserializeObject<List<AICreateEmailInfo>>(cleanJson);
  2892. }
  2893. catch (JsonException ex)
  2894. {
  2895. // 记录日志并考虑 fallback 策略
  2896. _logger.LogError(ex, "Hunyuan AI 响应解析失败。原始数据:{Response}", aiResponse);
  2897. }
  2898. }
  2899. if (hunyuanAIEmailDatas == null || !hunyuanAIEmailDatas.Any())
  2900. {
  2901. await HttpContext.SendSseStepAsync(-1, "AI 格式解析失败,请尝试重新生成");
  2902. return;
  2903. }
  2904. // 5. 更新数据模型 (Progress: 90%)
  2905. foreach (var client in clientInfoSources)
  2906. {
  2907. var aiEmail = hunyuanAIEmailDatas.FirstOrDefault(x => x.Guid == client.Guid);
  2908. if (aiEmail != null)
  2909. {
  2910. client.EmailInfo.Status = 2; // 已生成
  2911. client.EmailInfo.EmailTitle = aiEmail.Subject;
  2912. client.EmailInfo.EmailContent = aiEmail.Content;
  2913. client.EmailInfo.Operator = operatorName;
  2914. client.EmailInfo.OperatedAt = DateTime.Now;
  2915. }
  2916. }
  2917. // 6. 数据库持久化 (Progress: 95%)
  2918. // 采用增量更新策略,避免直接 Where 过滤掉其他未选中的数据
  2919. var update = await _sqlSugar.Updateable(invAiInfo).UpdateColumns(x => x.AiCrawledDetails).ExecuteCommandAsync();
  2920. if (update > 0)
  2921. {
  2922. await HttpContext.SendSseStepAsync(100, "邮件全部生成完毕并已存入团组资料库", invAiInfo.AiCrawledDetails);
  2923. }
  2924. else
  2925. {
  2926. await HttpContext.SendSseStepAsync(-1, "数据库写入失败,请联系管理员");
  2927. }
  2928. }
  2929. catch (Exception ex)
  2930. {
  2931. _logger.LogError(ex, "邮件生成异常");
  2932. await HttpContext.SendSseStepAsync(-1, $"生成失败:{ex.Message}");
  2933. }
  2934. finally
  2935. {
  2936. await HttpContext.FinalizeSseAsync();
  2937. }
  2938. }
  2939. /// <summary>
  2940. /// 混元邮件生成提示词构建
  2941. /// </summary>
  2942. /// <param name="originUnit">出访单位</param>
  2943. /// <param name="objective">出访室友/拜访目的</param>
  2944. /// <param name="visitDate">出访时间</param>
  2945. /// <param name="clientInfosForAI">需要生成邮件的客户信息列表</param>
  2946. /// <returns>最终构建的 System Prompt 字符串</returns>
  2947. private static string BuildHunyuanEmailPrompt(string originUnit, string objective,string visitDate, List<AICreateEmailInfo> clientInfosForAI)
  2948. {
  2949. return $@"
  2950. # Role
  2951. 你是一位精通国际政企关系的【首席联络官】。你具备极强的行业分析能力,能通过 [SourceEntity] 的名称自动检索并推导其行政职能与行业地位,并以此撰写具有战略高度、语调优雅的正式商务邮件。
  2952. # Intelligence Task: Source Profiling
  2953. 在生成邮件前,请先执行以下逻辑:
  2954. 1. **职能推导**:基于 [SourceEntity] 的名称,自动识别其在所属领域的具体行政职能与政策影响力。
  2955. 2. **战略对齐**:将其职能与全球大趋势(如:Sustainable Urbanization, Digital Transformation, Carbon Neutrality)挂钩,作为邮件第二段的叙事背景。
  2956. # [RICH_TEXT_STANDARDS_FOR_QUILL]
  2957. - **Prohibited Tags**: 绝对禁止包含 <html>, <body>, <head>, <!DOCTYPE>, <hr>, <style>。
  2958. - **Whitelisted Tags**: 仅允许使用 <h3> (用于主题), <p> (用于段落), <ul>, <li>, <strong>, <br>。
  2959. - **Structural Integrity**:
  2960. - 严禁出现裸露文本,所有段落必须由 <p> 包裹。
  2961. - 列表必须使用标准 <ul><li> 嵌套结构。
  2962. - **No Markdown**: 严禁在 Content 中出现 ###, **, --- 等 Markdown 符号。
  2963. # Reference Model (Few-Shot Example)
  2964. 在生成时,请严格参考以下范例的【语气】、【五段式结构】和【外交辞令】:
  2965. - Para 1: ""On behalf of [SourceEntity], I am writing to formally propose...""
  2966. - Para 2: ""As a pivotal megacity [AI根据Source地位补全]... we recognize [Target]’s leadership in...""
  2967. - Para 3: ""We are particularly keen to explore... [基于Target经营范围推导的3个技术点]...""
  2968. - Para 4: ""We propose a 2–3 day visit... during the week of [Logistics]...""
  2969. - Para 5: ""Enclosed please find the [PascalCase_File]...""
  2970. # Task
  2971. 根据 [TargetList] 中每个单位的【经营范围】,为 [SourceEntity] 生成独立的英文访问请求邮件。
  2972. # Inputs
  2973. - [SourceEntity]: [{originUnit}]
  2974. - [VisitPurpose]: [{objective}]
  2975. - [TargetList]: [{JsonConvert.SerializeObject(clientInfosForAI)}]
  2976. - [Logistics]: [{visitDate}]
  2977. # Execution Logic (Chain of Thought)
  2978. 1. **Scope-to-Focus Analysis**:
  2979. - 深入分析每个 Target 的【经营范围】。
  2980. - 自动推导 3 个与 [VisitPurpose] 高度对齐的专业考察点(如:Policy Frameworks, Technical Standards, Operational Case Studies)。
  2981. 2. **Modular Drafting**:
  2982. - 必须严格遵循参考范例的 5 段式逻辑。
  2983. - 针对不同职能属性(监管/技术/运营)动态微调邮件主题(Subject)。
  2984. 3. **No Personnel Reference**: 严禁提及“名单、人数、成员、审核中”等任何具体人员信息,保持机构对等对话的高度。
  2985. # Constraints & Standards
  2986. - **Tone**: Formal, Strategic, and Executive (庄重、具战略高度、执行力强)。
  2987. - **Naming Protocol**: 附件引用统一使用 `JointVisitAgenda`, `StrategicInquiryBrief` (PascalCase)。
  2988. - **Escaping**: 必须对 HTML 内部的所有双引号进行严格转义(\""),确保 JSON 字符串合法。
  2989. - **Output Format**: 仅输出一个合规的 JSON 数组,不包含任何解释性文字。
  2990. - **Field Mapping**:
  2991. - `Guid`: 原样保留。
  2992. - `NameCn`: 原样保留。
  2993. - `Scope`: 原样保留。
  2994. - `Subject`: 动态生成的英文主题。
  2995. - `Content`: 包含 QUILL的英文正文。
  2996. # JSON Structure Template
  2997. [
  2998. {{
  2999. ""Guid"": ""ID"",
  3000. ""NameCn"": ""Entity Name"",
  3001. ""Scope"": ""Original Scope Description"",
  3002. ""Subject"": ""Dynamic English Subject Line"",
  3003. ""Content"": ""<h3>Subject: ...</h3><p>Dear Leadership Team of <strong>[Target]</strong>,</p><p>On behalf of <strong>[SourceEntity]</strong>, I am writing to propose...</p><ul><li><strong>Focus Area:</strong> Technical analysis of...</li></ul><p>Sincerely,<br><strong>Office of the Chief Liaison</strong><br>[SourceEntity]</p>""
  3004. }}
  3005. ]";
  3006. }
  3007. /// <summary>
  3008. /// 商邀资料AI 邮件保存
  3009. /// </summary>
  3010. /// <returns></returns>
  3011. [HttpPost]
  3012. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  3013. public async Task<IActionResult> InvitationAIEmailSave(InvitationAIEmailSaveDto dto)
  3014. {
  3015. if (dto.Id < 1) return Ok(JsonView(false, "请选择保存的团组"));
  3016. if (dto.CurrUserId < 1) return Ok(JsonView(false, "请传入用户Id"));
  3017. if (string.IsNullOrEmpty(dto.Guid)) return Ok(JsonView(false, "请传入Guid"));
  3018. if (string.IsNullOrEmpty(dto.EmailTitle)) return Ok(JsonView(false, "请传入邮件标题"));
  3019. if (string.IsNullOrEmpty(dto.EmailContent)) return Ok(JsonView(false, "请传入邮件内容"));
  3020. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI>()
  3021. .Where(x => x.Id == dto.Id)
  3022. .FirstAsync();
  3023. if (invAiInfo == null) return Ok(JsonView(false, "数据信息为空"));
  3024. var dataList = invAiInfo?.AiCrawledDetails;
  3025. if (dataList == null) return Ok(JsonView(false, "邀请方数据信息为空"));
  3026. var editInfo = dataList.Where(x => x.Guid == dto.Guid).FirstOrDefault();
  3027. if (editInfo == null) return Ok(JsonView(false, "邀请方信息为空"));
  3028. var opUserName = await _sqlSugar.Queryable<Sys_Users>().Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  3029. editInfo.EmailInfo.EmailTitle = dto.EmailTitle;
  3030. editInfo.EmailInfo.EmailContent = dto.EmailContent;
  3031. editInfo.EmailInfo.Operator = opUserName;
  3032. editInfo.EmailInfo.OperatedAt = DateTime.Now;
  3033. var datas = dataList.Where(x => x.Guid != dto.Guid).ToList();
  3034. datas.Add(editInfo);
  3035. // 排序
  3036. invAiInfo.AiCrawledDetails = datas.OrderByDescending(x => x.OperatedAt).ToList();
  3037. var editUpd = await _sqlSugar.Updateable(invAiInfo).UpdateColumns(x => x.AiCrawledDetails).ExecuteCommandAsync();
  3038. if (editUpd < 1)
  3039. {
  3040. return Ok(JsonView(true, "邮件保存失败"));
  3041. }
  3042. return Ok(JsonView(true, "邮件保存成功"));
  3043. }
  3044. /// <summary>
  3045. /// 商邀资料AI 邮箱附件上传
  3046. /// </summary>
  3047. [HttpPost]
  3048. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  3049. public async Task<IActionResult> InvitationAIFileSave([FromForm] InvitationAIFileSaveDto dto)
  3050. {
  3051. // 1. 炼金前置:严格参数校验
  3052. if (dto.Id < 1) return Ok(JsonView(false, "请选择保存的团组"));
  3053. if (dto.CurrUserId < 1) return Ok(JsonView(false, "请传入用户Id"));
  3054. if (string.IsNullOrEmpty(dto.Guid)) return Ok(JsonView(false, "请传入Guid"));
  3055. if (dto.Attachments == null || !dto.Attachments.Any()) return Ok(JsonView(false, "请传入附件"));
  3056. // 2. 数据获取:利用 SqlSugar 异步查询
  3057. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI>()
  3058. .InSingleAsync(dto.Id);
  3059. if (invAiInfo?.AiCrawledDetails == null) return Ok(JsonView(false, "邀请方信息不存在"));
  3060. // 3. 业务定位:定位到具体需要编辑的 Guid 记录
  3061. var editInfo = invAiInfo.AiCrawledDetails.FirstOrDefault(x => x.Guid == dto.Guid);
  3062. if (editInfo == null) return Ok(JsonView(false, "未找到匹配的 Guid 记录"));
  3063. // 4. 追踪信息更新
  3064. var opUserName = await _sqlSugar.Queryable<Sys_Users>()
  3065. .Where(x => x.Id == dto.CurrUserId && x.IsDel == 0)
  3066. .Select(x => x.CnName)
  3067. .FirstAsync() ?? "-";
  3068. // 更新操作人与时间 (利用引用类型特性)
  3069. editInfo.Operator = opUserName;
  3070. editInfo.OperatedAt = DateTime.Now;
  3071. // 初始化 EmailInfo 确保不为 null
  3072. editInfo.EmailInfo ??= new EmailInfo();
  3073. editInfo.EmailInfo.Operator = opUserName;
  3074. editInfo.EmailInfo.OperatedAt = DateTime.Now;
  3075. editInfo.EmailInfo.AttachmentPaths ??= new List<string>();
  3076. // 5. 构建绝对路径与相对路径
  3077. string dirName = $"{editInfo.NameEn?.Trim() ?? "Default"}_{dto.Guid}";
  3078. string baseDir = AppSettingsHelper.Get("InvitationAIAssistBasePath");
  3079. string ftpBase = AppSettingsHelper.Get("InvitationAIAssistFtpPath");
  3080. string absolutePath = Path.Combine(baseDir, dirName);
  3081. string relativePathPrefix = $"{ftpBase}{dirName}/";
  3082. // 核心修复:确保父级目录递归创建,防止 DirectoryNotFoundException
  3083. if (!Directory.Exists(absolutePath))
  3084. {
  3085. Directory.CreateDirectory(absolutePath);
  3086. }
  3087. var newSavedRelativePaths = new List<string>();
  3088. // 6. 持续文件存储与实时验证
  3089. foreach (var file in dto.Attachments)
  3090. {
  3091. // 文件名安全过滤
  3092. string safeName = string.Join("_", file.FileName.Split(Path.GetInvalidFileNameChars()));
  3093. string fullPath = Path.Combine(absolutePath, safeName);
  3094. // 重名冲突处理:文件名(n).ext
  3095. if (System.IO.File.Exists(fullPath))
  3096. {
  3097. string fileNameOnly = Path.GetFileNameWithoutExtension(safeName);
  3098. string extension = Path.GetExtension(safeName);
  3099. int count = 1;
  3100. while (System.IO.File.Exists(fullPath))
  3101. {
  3102. safeName = $"{fileNameOnly}({count++}){extension}";
  3103. fullPath = Path.Combine(absolutePath, safeName);
  3104. }
  3105. }
  3106. try
  3107. {
  3108. // 写入流:使用作用域隔离,确保退出 using 时立即释放句柄
  3109. using (var stream = new FileStream(fullPath, FileMode.Create, FileAccess.Write, FileShare.None))
  3110. {
  3111. await file.CopyToAsync(stream);
  3112. await stream.FlushAsync(); // 强制落盘
  3113. }
  3114. // 实时验证:物理验证 + 逻辑校验
  3115. var fileInfo = new FileInfo(fullPath);
  3116. if (fileInfo.Exists && fileInfo.Length == file.Length)
  3117. {
  3118. newSavedRelativePaths.Add($"{relativePathPrefix}{safeName}");
  3119. }
  3120. }
  3121. catch (Exception ex)
  3122. {
  3123. // 发生 IO 异常时清理残余文件并抛出,触发外层处理
  3124. if (System.IO.File.Exists(fullPath)) System.IO.File.Delete(fullPath);
  3125. // 此处记录日志:
  3126. _logger.LogError(ex, "File Save Error");
  3127. continue; // 跳过失败的文件,继续下一个
  3128. }
  3129. }
  3130. if (!newSavedRelativePaths.Any()) return Ok(JsonView(false, "所有文件上传均失败"));
  3131. // 7. 数据合并与持久化:
  3132. // 将新路径与旧路径合并,并去重
  3133. var updatedPaths = editInfo.EmailInfo.AttachmentPaths;
  3134. updatedPaths.AddRange(newSavedRelativePaths);
  3135. editInfo.EmailInfo.AttachmentPaths = updatedPaths.Distinct().ToList();
  3136. // 重新排序整体列表
  3137. invAiInfo.AiCrawledDetails = invAiInfo.AiCrawledDetails
  3138. .OrderByDescending(x => x.OperatedAt)
  3139. .ToList();
  3140. // SqlSugar 高效部分列更新
  3141. var isSuccess = await _sqlSugar.Updateable(invAiInfo)
  3142. .UpdateColumns(x => x.AiCrawledDetails)
  3143. .ExecuteCommandAsync() > 0;
  3144. if (!isSuccess) return Ok(JsonView(false, "数据库更新失败"));
  3145. // 8. 返回前端数据 (支持 lowerCamelCase)
  3146. var officeBaseUrl = AppSettingsHelper.Get("OfficeBaseUrl");
  3147. var finalResultUrls = editInfo.EmailInfo.AttachmentPaths
  3148. .Select(path => $"{officeBaseUrl}{path}")
  3149. .ToList();
  3150. return Ok(JsonView(true, "邮件附件保存成功", finalResultUrls));
  3151. }
  3152. /// <summary>
  3153. /// 商邀资料AI 邮箱附件删除
  3154. /// </summary>
  3155. [HttpPost]
  3156. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  3157. public async Task<IActionResult> InvitationAIFileDel([FromBody] InvitationAIFileDelDto dto)
  3158. {
  3159. // 1. 基础校验
  3160. if (dto.Id < 1) return Ok(JsonView(false, "请选择团组"));
  3161. if (string.IsNullOrEmpty(dto.Guid)) return Ok(JsonView(false, "请传入Guid"));
  3162. if (dto.AttachmentNames == null || !dto.AttachmentNames.Any()) return Ok(JsonView(false, "请传入待删除附件名称"));
  3163. // 2. 数据获取
  3164. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI>().InSingleAsync(dto.Id);
  3165. if (invAiInfo?.AiCrawledDetails == null) return Ok(JsonView(false, "业务数据不存在"));
  3166. var editInfo = invAiInfo.AiCrawledDetails.FirstOrDefault(x => x.Guid == dto.Guid);
  3167. if (editInfo?.EmailInfo?.AttachmentPaths == null) return Ok(JsonView(false, "未找到对应的附件记录"));
  3168. // 3. 操作人追踪
  3169. var opUserName = await _sqlSugar.Queryable<Sys_Users>()
  3170. .Where(x => x.Id == dto.CurrUserId && x.IsDel == 0)
  3171. .Select(x => x.CnName).FirstAsync() ?? "System";
  3172. // 更新追踪状态
  3173. editInfo.Operator = editInfo.EmailInfo.Operator = opUserName;
  3174. editInfo.OperatedAt = editInfo.EmailInfo.OperatedAt = DateTime.Now;
  3175. // 4. 路径
  3176. string dirName = $"{editInfo.NameEn?.Trim() ?? "Default"}_{dto.Guid}";
  3177. string absoluteDir = Path.Combine(AppSettingsHelper.Get("InvitationAIAssistBasePath"), dirName);
  3178. // 获取当前数据库中的相对路径列表
  3179. var currentPaths = editInfo.EmailInfo.AttachmentPaths;
  3180. bool isChanged = false;
  3181. // 5. 核心删除逻辑
  3182. foreach (var fileName in dto.AttachmentNames)
  3183. {
  3184. // 查找数据库中是否存在包含该文件名的路径 (忽略大小写比较)
  3185. var targetRelativePath = currentPaths.FirstOrDefault(p => p.EndsWith("/" + fileName) || p.Equals(fileName));
  3186. if (targetRelativePath != null)
  3187. {
  3188. // A. 从数据库记录中移除
  3189. currentPaths.Remove(targetRelativePath);
  3190. isChanged = true;
  3191. // B. 物理文件删除逻辑
  3192. // 注意:fullPath 必须是 [基础路径] + [目录名] + [纯文件名]
  3193. string fullPath = Path.Combine(absoluteDir, fileName);
  3194. try
  3195. {
  3196. if (System.IO.File.Exists(fullPath))
  3197. {
  3198. System.IO.File.Delete(fullPath);
  3199. }
  3200. }
  3201. catch (Exception ex)
  3202. {
  3203. // 记录 IO 异常但不中断流程
  3204. _logger.LogWarning($"物理文件删除失败: {fullPath}, {ex.Message}");
  3205. }
  3206. }
  3207. }
  3208. if (!isChanged) return Ok(JsonView(false, "未找到匹配的可删除附件"));
  3209. // 6. 数据同步与持久化
  3210. editInfo.EmailInfo.AttachmentPaths = currentPaths;
  3211. // 排序 (利用引用类型,无需手动 Add/Remove)
  3212. invAiInfo.AiCrawledDetails = invAiInfo.AiCrawledDetails.OrderByDescending(x => x.OperatedAt).ToList();
  3213. var isUpd = await _sqlSugar.Updateable(invAiInfo)
  3214. .UpdateColumns(x => x.AiCrawledDetails)
  3215. .ExecuteCommandAsync() > 0;
  3216. if (!isUpd) return Ok(JsonView(false, "数据库记录更新失败"));
  3217. // 7. 返回剩余附件的完整访问地址
  3218. var officeBaseUrl = AppSettingsHelper.Get("OfficeBaseUrl");
  3219. var remainingUrls = editInfo.EmailInfo.AttachmentPaths
  3220. .Select(path => $"{officeBaseUrl}{path}")
  3221. .ToList();
  3222. return Ok(JsonView(true, "附件删除成功", remainingUrls));
  3223. }
  3224. /// <summary>
  3225. /// 商邀资料AI 发送邮件(SSE 流式推送)
  3226. /// </summary>
  3227. [HttpPost]
  3228. public async Task InvitationAISeedEmailStream([FromBody] InvitationAISeedEmailDto dto)
  3229. {
  3230. // 1. 初始化 SSE
  3231. HttpContext.InitializeSse();
  3232. try
  3233. {
  3234. await HttpContext.SendSseStepAsync(5, "正在准备发送邮件队列...");
  3235. #region 1. 参数与权限前置校验
  3236. if (dto.Id < 1 || dto.CurrUserId < 1 || dto.Guids == null || !dto.Guids.Any())
  3237. {
  3238. await HttpContext.SendSseStepAsync(-1, "参数验证失败,请检查选择的单位及用户状态。");
  3239. return;
  3240. }
  3241. // hotmail 配置信息验证
  3242. var hotmailConfig = await _hotmailService.GetUserMailConfig(dto.CurrUserId);
  3243. (bool verify,string msg) = _hotmailService.ConfigVerify(hotmailConfig);
  3244. if (!verify)
  3245. {
  3246. await HttpContext.SendSseStepAsync(-1, msg);
  3247. return;
  3248. }
  3249. // 获取商邀信息和用户信息
  3250. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI>().FirstAsync(x => x.IsDel == 0 && x.Id == dto.Id);
  3251. var userInfo = await _sqlSugar.Queryable<Sys_Users>()
  3252. .Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId)
  3253. .Select(x => new { x.Email, x.CnName })
  3254. .FirstAsync();
  3255. if (invAiInfo?.AiCrawledDetails == null)
  3256. {
  3257. await HttpContext.SendSseStepAsync(-1, "未找到有效的邀请方数据。");
  3258. return;
  3259. }
  3260. // 提取待发送的目标集合
  3261. var guidSet = new HashSet<string>(dto.Guids);
  3262. var seedInvInfos = invAiInfo.AiCrawledDetails.Where(x => guidSet.Contains(x.Guid)).ToList();
  3263. if (!seedInvInfos.Any())
  3264. {
  3265. await HttpContext.SendSseStepAsync(-1, "所选单位信息在原始库中不存在。");
  3266. return;
  3267. }
  3268. #endregion
  3269. await HttpContext.SendSseStepAsync(15, $"准备就绪,共计 {seedInvInfos.Count} 封邮件待发送...");
  3270. #region 2. 批量发送逻辑 (流式反馈)
  3271. int total = seedInvInfos.Count;
  3272. int current = 0;
  3273. var successCount = 0;
  3274. var failCount = 0;
  3275. foreach (var item in seedInvInfos)
  3276. {
  3277. current++;
  3278. // 计算进度:从 20% 到 90%
  3279. int progress = 20 + (int)((double)current / total * 70);
  3280. if (string.IsNullOrEmpty(item.EmailInfo?.EmailTitle) || string.IsNullOrEmpty(item.EmailInfo?.EmailContent))
  3281. {
  3282. await HttpContext.SendSseStepAsync(progress, $"跳过:{item.NameCn} (邮件标题或内容缺失)");
  3283. failCount++;
  3284. continue;
  3285. }
  3286. try
  3287. {
  3288. var req = new MailDto() {
  3289. Subject = item.EmailInfo.EmailTitle,
  3290. Content = item.EmailInfo.EmailContent,
  3291. To = item.Email,
  3292. AttachmentPaths = item.EmailInfo.AttachmentPaths
  3293. };
  3294. var res = await _hotmailService.SendMailAsync(hotmailConfig.UserName, req);
  3295. if (res.IsSuccess)
  3296. {
  3297. successCount++;
  3298. // 更新本地状态
  3299. item.EmailInfo.Status = 4; // 发送成功状态
  3300. item.EmailInfo.Operator = userInfo.CnName;
  3301. item.EmailInfo.OperatedAt = DateTime.Now;
  3302. await HttpContext.SendSseStepAsync(progress, $"成功:已向 {item.NameCn}({item.Email}) 发送邮件");
  3303. }
  3304. else
  3305. {
  3306. failCount++;
  3307. await HttpContext.SendSseStepAsync(progress, $"失败:{item.NameCn} 发送失败 ({res.Message})");
  3308. }
  3309. }
  3310. catch (Exception ex)
  3311. {
  3312. failCount++;
  3313. _logger.LogError(ex, $"企微邮件推送异常:{item.NameCn}");
  3314. await HttpContext.SendSseStepAsync(progress, $"异常:{item.NameCn} 连接超时");
  3315. }
  3316. }
  3317. #endregion
  3318. await HttpContext.SendSseStepAsync(95, "正在归档发送记录...");
  3319. #region 3. 数据同步与收尾
  3320. // 更新内存中的全量数据:排除旧的,加入已更新状态的
  3321. invAiInfo.AiCrawledDetails = invAiInfo.AiCrawledDetails
  3322. .Where(x => !guidSet.Contains(x.Guid))
  3323. .Concat(seedInvInfos)
  3324. .OrderByDescending(x => x.EmailInfo.OperatedAt)
  3325. .ToList();
  3326. // 精准更新 JSON 列
  3327. var updateOk = await _sqlSugar.Updateable(invAiInfo)
  3328. .UpdateColumns(x => x.AiCrawledDetails)
  3329. .ExecuteCommandHasChangeAsync();
  3330. if (!updateOk)
  3331. {
  3332. await HttpContext.SendSseStepAsync(-1, "记录保存失败,请检查数据库连接。");
  3333. return;
  3334. }
  3335. await HttpContext.SendSseStepAsync(100, $"任务完成!成功: {successCount}, 失败: {failCount}", new
  3336. {
  3337. Id = invAiInfo.Id,
  3338. SuccessCount = successCount,
  3339. FailCount = failCount
  3340. });
  3341. #endregion
  3342. }
  3343. catch (Exception ex)
  3344. {
  3345. _logger.LogError(ex, "邮件发送流发生崩溃");
  3346. await HttpContext.SendSseStepAsync(-1, $"系统错误:{ex.Message}");
  3347. }
  3348. finally
  3349. {
  3350. await HttpContext.FinalizeSseAsync();
  3351. }
  3352. }
  3353. #endregion
  3354. #region 商邀资料 AI - 无团组版
  3355. /// <summary>
  3356. /// 商邀资料AI-无团组版 基础数据源
  3357. /// </summary>
  3358. /// <returns></returns>
  3359. [HttpGet]
  3360. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  3361. public async Task<IActionResult> InvitationAI_NoGroupInit()
  3362. {
  3363. var itemNames = await InvitationAI_NoGroupInvName();
  3364. var unitNames = await InvitationAIClientName();
  3365. var countries = await InvitationAICountryName();
  3366. var industryNodes = IndustryTree.Build().Select(x => x.NameCn).ToList();
  3367. var orgScales = OrgScale.BuildInitialData().Select(x => x.Name).ToList();
  3368. var orgLevels = new List<string>() {
  3369. "全部",
  3370. "总部",
  3371. "分公司"
  3372. };
  3373. return Ok(JsonView(true, $"查询成功!", new
  3374. {
  3375. itemNames,
  3376. unitNames,
  3377. countries,
  3378. industryNodes,
  3379. orgScales,
  3380. orgLevels
  3381. }));
  3382. }
  3383. /// <summary>
  3384. /// 商邀资料AI-无团组版 资料列表
  3385. /// </summary>
  3386. /// <returns></returns>
  3387. [HttpGet("{name}")]
  3388. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  3389. public async Task<IActionResult> InvitationAI_NoGroupItemByName(string name)
  3390. {
  3391. if (string.IsNullOrWhiteSpace(name)) return Ok(JsonView(false, "名称无效"));
  3392. // 1. 获取主表记录
  3393. var info = await _sqlSugar.Queryable<Res_InvitationAI_NoGroup>()
  3394. .FirstAsync(x => x.IsDel == 0 && x.InvName == name);
  3395. // 2. 分支处理:若无主表记录,尝试从团组表同步基本信息
  3396. if (info == null)
  3397. {
  3398. return Ok(JsonView(true, "暂无数据", new
  3399. {
  3400. Id = 0,
  3401. InvName = name,
  3402. AiCrawledDetails = new List<InvitationAI_NoGroupInfo>(),
  3403. }));
  3404. }
  3405. // 3. 数据填充(使用 ??= 语法确保 Null 安全)
  3406. info.EntryInfo ??= new Entry_NoGroupInfo();
  3407. // 4. 排序:ThenByDescending 确保多级排序生效
  3408. info.AiCrawledDetails = info.AiCrawledDetails
  3409. .OrderByDescending(x => x.IsChecked)
  3410. .ThenByDescending(x => x.OperatedAt)
  3411. .ToList();
  3412. var details = info?.AiCrawledDetails ?? new List<InvitationAI_NoGroupInfo>();
  3413. string baseUrl = AppSettingsHelper.Get("OfficeBaseUrl")?.TrimEnd('/');
  3414. if (details.Any())
  3415. {
  3416. foreach (var detail in details)
  3417. {
  3418. // 附件路径处理
  3419. if (detail.EmailInfo?.AttachmentPaths?.Any() == true)
  3420. {
  3421. detail.EmailInfo.AttachmentPaths = detail.EmailInfo.AttachmentPaths
  3422. .Select(p => p.StartsWith("http") ? p : $"{baseUrl}/{p.TrimStart('/')}")
  3423. .ToList();
  3424. }
  3425. // 官网地址注入新闻首位 (确保 PostUrl 已实例化)
  3426. if (!string.IsNullOrEmpty(detail.SiteUrl))
  3427. {
  3428. detail.PostUrl ??= new List<PostNewsItem>();
  3429. // 避免重复插入
  3430. if (!detail.PostUrl.Any(x => x.Description == "官网"))
  3431. {
  3432. detail.PostUrl.Insert(0, new PostNewsItem
  3433. {
  3434. Date = "-",
  3435. Description = "官网",
  3436. Url = detail.SiteUrl
  3437. });
  3438. }
  3439. }
  3440. }
  3441. }
  3442. return Ok(JsonView(true, "查询成功!", new
  3443. {
  3444. info.Id,
  3445. info.InvName,
  3446. AiCrawledDetails = details,
  3447. Entry = info.EntryInfo
  3448. }));
  3449. }
  3450. /// <summary>
  3451. /// 商邀资料AI-无团组版 新增公务资料
  3452. /// </summary>
  3453. /// <returns></returns>
  3454. [HttpPost]
  3455. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  3456. public async Task<IActionResult> InvitationAI_NoGroupAdd(InvitationAI_NoGroupAddDto dto)
  3457. {
  3458. // 基础校验
  3459. if (string.IsNullOrWhiteSpace(dto.InvName))
  3460. return Ok(JsonView(false, "请传入有效的公务名称!"));
  3461. if (dto.CurrUserId <= 0)
  3462. return Ok(JsonView(false, "用户身份无效"));
  3463. var invName = dto.InvName;
  3464. #region 数据库操作
  3465. // 数据库信息获取方式
  3466. var invExist = await _sqlSugar.Queryable<Res_InvitationAI_NoGroup>().AnyAsync(x => x.IsDel == 0 && x.InvName == invName);
  3467. if (invExist)
  3468. return Ok(JsonView(false, $"公务名称 [{invName}] 已存在,请更换"));
  3469. string operatorName = await _sqlSugar.Queryable<Sys_Users>().Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  3470. var dataInfo = new Res_InvitationAI_NoGroup()
  3471. {
  3472. InvName = invName,
  3473. EntryInfo = new Entry_NoGroupInfo()
  3474. {
  3475. Operator = operatorName,
  3476. OperatedAt = DateTime.Now
  3477. },
  3478. CreateUserId = dto.CurrUserId
  3479. };
  3480. var insert = await _sqlSugar.Insertable(dataInfo).ExecuteReturnIdentityAsync();
  3481. if (insert < 1)
  3482. {
  3483. return Ok(JsonView(false, $"词条信息新增失败!"));
  3484. }
  3485. #endregion
  3486. return Ok(JsonView(true, $"新增成功!", new
  3487. {
  3488. id = insert,
  3489. invName = dataInfo.InvName,
  3490. entry = dataInfo.EntryInfo,
  3491. }));
  3492. }
  3493. /// <summary>
  3494. /// 商邀资料AI-无团组版 设置词条
  3495. /// </summary>
  3496. /// <returns></returns>
  3497. [HttpPost]
  3498. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  3499. public async Task<IActionResult> InvitationAI_NoGroupSetPrompt(InvitationAI_NoGroupSetPromptDto dto)
  3500. {
  3501. if (dto.Id < 1) return Ok(JsonView(false, "当前公务ID无效"));
  3502. if (string.IsNullOrWhiteSpace(dto.Objective)) return Ok(JsonView(false, "请输入出访目的"));
  3503. if (string.IsNullOrWhiteSpace(dto.OriginUnit)) return Ok(JsonView(false, "请输入单位名称"));
  3504. if (dto.TargetCountry == null || !dto.TargetCountry.Any()) return Ok(JsonView(false, "请选择目标国家"));
  3505. if (dto.Industries == null || !dto.Industries.Any()) return Ok(JsonView(false, "请选择所属行业"));
  3506. if (dto.ScaleTypes == null || !dto.ScaleTypes.Any()) return Ok(JsonView(false, "请选择规模类型"));
  3507. if (string.IsNullOrWhiteSpace(dto.VisitDate)) return Ok(JsonView(false, "请输入出访时间"));
  3508. // 数据库信息获取方式
  3509. var dataInfo = await _sqlSugar.Queryable<Res_InvitationAI_NoGroup>().Where(x => x.IsDel == 0 && x.Id == dto.Id).FirstAsync();
  3510. if (dataInfo == null)
  3511. return Ok(JsonView(false, "请先新增公务信息"));
  3512. string operatorName = await _sqlSugar.Queryable<Sys_Users>().Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  3513. dataInfo.EntryInfo = new Entry_NoGroupInfo()
  3514. {
  3515. VisitDate = dto.VisitDate?.Trim(),
  3516. OriginUnit = dto.OriginUnit?.Trim(),
  3517. TargetCountry = dto.TargetCountry,
  3518. Objective = dto.Objective?.Trim(),
  3519. Industries = dto.Industries,
  3520. ScaleTypes = dto.ScaleTypes,
  3521. IsBackground = dto.IsBackground,
  3522. OrgLevel = dto.OrgLevel,
  3523. OtherConstraints = dto.OtherConstraints,
  3524. Operator = operatorName,
  3525. OperatedAt = DateTime.Now
  3526. };
  3527. var updateResult = await _sqlSugar.Updateable(dataInfo)
  3528. .UpdateColumns(x => new { x.EntryInfo })
  3529. .ExecuteCommandHasChangeAsync();
  3530. if (!updateResult)
  3531. {
  3532. return Ok(JsonView(false, $"词条信息更新失败!"));
  3533. }
  3534. return Ok(JsonView(true, $"设置成功!", new
  3535. {
  3536. id = dataInfo.Id,
  3537. invName = dataInfo.InvName,
  3538. entry = dataInfo.EntryInfo
  3539. }));
  3540. }
  3541. /// <summary>
  3542. /// 商邀资料AI-无团组版 设置复选框选中 批量(支持传入空数组进行重置)
  3543. /// </summary>
  3544. /// <param name="dto"></param>
  3545. /// <returns></returns>
  3546. [HttpPost]
  3547. public async Task<IActionResult> InvitationAI_NoGroupSetChecked([FromBody] InvitationAI_NoGroupSetCheckedDto dto)
  3548. {
  3549. // 1. 基础参数校验 (注意:这里移除了对 Guids.Any() 的强校验,允许空集合)
  3550. if (dto.Id < 1 || dto.CurrUserId < 1 || dto.Guids == null)
  3551. return Ok(JsonView(false, "请求参数不完整"));
  3552. // 2. 获取主记录
  3553. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI_NoGroup>()
  3554. .FirstAsync(x => x.IsDel == 0 && x.Id == dto.Id);
  3555. if (invAiInfo?.AiCrawledDetails == null || !invAiInfo.AiCrawledDetails.Any())
  3556. return Ok(JsonView(false, "数据不存在或集合为空"));
  3557. // 3. 准备更新所需数据
  3558. var guidSet = dto.Guids.ToHashSet();
  3559. var now = DateTime.Now;
  3560. // 获取操作人姓名
  3561. string operatorName = await _sqlSugar.Queryable<Sys_Users>()
  3562. .Where(x => x.Id == dto.CurrUserId)
  3563. .Select(x => x.CnName)
  3564. .FirstAsync() ?? "-";
  3565. // 4. 单次遍历完成 重置 + 更新
  3566. // 如果 guidSet 为空,循环会将所有项的 IsChecked 置为 false
  3567. foreach (var item in invAiInfo.AiCrawledDetails)
  3568. {
  3569. if (guidSet.Contains(item.Guid))
  3570. {
  3571. item.IsChecked = true;
  3572. }
  3573. else
  3574. {
  3575. item.IsChecked = false;
  3576. }
  3577. item.OperatedAt = now;
  3578. item.Operator = operatorName;
  3579. }
  3580. // 5. 排序逻辑
  3581. // 即使是全量取消选中,也可以按最后操作时间排序,让最近变动的项在前
  3582. invAiInfo.AiCrawledDetails = invAiInfo.AiCrawledDetails
  3583. .OrderByDescending(x => x.OperatedAt)
  3584. .ToList();
  3585. // 6. 提交数据库
  3586. // 注意:SqlSugar 的 UpdateColumns 会序列化整个对象列表并覆盖数据库字段
  3587. await _sqlSugar.Updateable(invAiInfo)
  3588. .UpdateColumns(x => x.AiCrawledDetails)
  3589. .ExecuteCommandAsync();
  3590. return Ok(JsonView(true, guidSet.Any() ? "设置成功" : "已全部取消选中"));
  3591. }
  3592. /// <summary>
  3593. /// 商邀资料AI-无团组版 保存
  3594. /// </summary>
  3595. /// <returns></returns>
  3596. [HttpPost]
  3597. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  3598. public async Task<IActionResult> InvitationAI_NoGroupSave(InvitationAI_NoGroupSaveDto dto)
  3599. {
  3600. if (dto.Id < 1) return Ok(JsonView(false, "请先设置AI关键字"));
  3601. if (dto.CurrUserId < 1) return Ok(JsonView(false, "请传入用户Id"));
  3602. if (dto.AiCrawledDetail == null) return Ok(JsonView(false, "请选择保存的邀请方详情"));
  3603. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI_NoGroup>()
  3604. .Where(x => x.Id == dto.Id)
  3605. .FirstAsync();
  3606. if (invAiInfo == null) return Ok(JsonView(false, "数据信息为空"));
  3607. var dataList = invAiInfo?.AiCrawledDetails;
  3608. if (dataList == null) return Ok(JsonView(false, "邀请方数据信息为空"));
  3609. var datas = dataList.Where(x => x.Guid != dto.AiCrawledDetail.Guid).ToList();
  3610. var editInfo = dto.AiCrawledDetail;
  3611. // 如果 Guid 为空,说明是新增数据,需要生成新的 Guid
  3612. if (string.IsNullOrEmpty(editInfo.Guid))
  3613. {
  3614. editInfo.Guid = Guid.NewGuid().ToString("N");
  3615. editInfo.Source = 2; // 标记为手动新增数据
  3616. editInfo.OperatedAt = DateTime.Now;
  3617. }
  3618. var opUserName = await _sqlSugar.Queryable<Sys_Users>().Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  3619. editInfo.Operator = opUserName;
  3620. editInfo.OperatedAt = DateTime.Now;
  3621. datas.Add(editInfo);
  3622. invAiInfo.AiCrawledDetails = datas.OrderByDescending(x => x.OperatedAt).ToList();
  3623. var editUpd = await _sqlSugar.Updateable(invAiInfo).UpdateColumns(x => x.AiCrawledDetails).ExecuteCommandAsync();
  3624. if (editUpd < 1)
  3625. {
  3626. return Ok(JsonView(true, "保存失败"));
  3627. }
  3628. return Ok(JsonView(true, "保存成功"));
  3629. }
  3630. /// <summary>
  3631. /// 商邀资料AI-无团组版 单条删除
  3632. /// </summary>
  3633. /// <returns></returns>
  3634. [HttpPost]
  3635. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  3636. public async Task<IActionResult> InvitationAI_NoGroupSingleDel(InvitationAI_NoGroupSingleDelDto dto)
  3637. {
  3638. int id = dto.Id;
  3639. string guid = dto.Guid;
  3640. // 基础校验
  3641. if (id < 1 || string.IsNullOrWhiteSpace(guid))
  3642. return Ok(JsonView(false, "请传入有效的Id和Guid!"));
  3643. var dataInfo = await _sqlSugar.Queryable<Res_InvitationAI_NoGroup>().Where(x => x.IsDel == 0 && x.Id == id).FirstAsync();
  3644. if (dataInfo == null)
  3645. return Ok(JsonView(false, "当前数据信息不存在!"));
  3646. var newDataInfos = dataInfo.AiCrawledDetails.Where(x => x.Guid != guid).ToList();
  3647. dataInfo.AiCrawledDetails = newDataInfos.OrderByDescending(x => x.OperatedAt).ToList();
  3648. var update = await _sqlSugar.Updateable(dataInfo).ExecuteCommandAsync();
  3649. if (update < 1)
  3650. {
  3651. return Ok(JsonView(false, $"删除失败!"));
  3652. }
  3653. return Ok(JsonView(true, $"删除成功!", new
  3654. {
  3655. dataInfo.Id,
  3656. dataInfo.InvName,
  3657. dataInfo.AiCrawledDetails
  3658. }));
  3659. }
  3660. /// <summary>
  3661. /// 商邀资料AI-无团组版 AI查询出的数据添加至商邀资料库
  3662. /// </summary>
  3663. /// <returns></returns>
  3664. [HttpPost]
  3665. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  3666. public async Task<IActionResult> InvitationAI_NoGroupInsertResource([FromBody] InvitationAI_NoGroupInsertResourceDto dto)
  3667. {
  3668. // 基础校验
  3669. if (dto.Id < 1 || dto.CurrUserId < 1 || dto.Guids == null || !dto.Guids.Any())
  3670. return Ok(JsonView(false, "参数不完整,请选择有效的单位数据"));
  3671. // 获取主记录
  3672. var dataInfo = await _sqlSugar.Queryable<Res_InvitationAI_NoGroup>()
  3673. .FirstAsync(x => x.IsDel == 0 && x.Id == dto.Id);
  3674. if (dataInfo?.AiCrawledDetails == null)
  3675. return Ok(JsonView(false, "当前数据信息不存在!"));
  3676. // 筛选出待转正的 AI 数据 (使用 HashSet 优化匹配)
  3677. var guidSet = dto.Guids.ToHashSet();
  3678. var targetAiDetails = dataInfo.AiCrawledDetails
  3679. .Where(x => x.Source != 0 && guidSet.Contains(x.Guid))
  3680. .ToList();
  3681. if (!targetAiDetails.Any())
  3682. return Ok(JsonView(false, "未找到符合条件的 AI 获取信息,无法重复添加或来源错误!"));
  3683. // 准备入库数据(批量构建)
  3684. var now = DateTime.Now;
  3685. var opUserName = await _sqlSugar.Queryable<Sys_Users>().Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  3686. var rawLocalDatas = targetAiDetails.Select(item => {
  3687. // 修改内存中的状态为本地数据
  3688. item.Source = 0;
  3689. item.OperatedAt = now;
  3690. item.Operator = opUserName;
  3691. return new Res_InvitationOfficialActivityData()
  3692. {
  3693. // 批量加密操作
  3694. Country = AesEncryptionHelper.Encrypt(item.Region),
  3695. UnitName = AesEncryptionHelper.Encrypt(item.NameCn),
  3696. Address = AesEncryptionHelper.Encrypt(item.Address),
  3697. Field = AesEncryptionHelper.Encrypt(item.Scope),
  3698. Contact = AesEncryptionHelper.Encrypt(item.Contact),
  3699. Tel = AesEncryptionHelper.Encrypt(item.Phone),
  3700. Email = AesEncryptionHelper.Encrypt(item.Email),
  3701. UnitWeb = AesEncryptionHelper.Encrypt(item.SiteUrl),
  3702. LastUpdateUserId = dto.CurrUserId,
  3703. LastUpdateTime = now,
  3704. CreateUserId = dto.CurrUserId,
  3705. CreateTime = now
  3706. };
  3707. }).ToList();
  3708. // 开启事务执行双表更新
  3709. var result = await _sqlSugar.UseTranAsync(async () =>
  3710. {
  3711. // 插入本地资料库
  3712. await _sqlSugar.Insertable(rawLocalDatas).ExecuteCommandAsync();
  3713. // 更新 AI 主表的状态(全量覆盖 JSON 列)
  3714. await _sqlSugar.Updateable(dataInfo)
  3715. .UpdateColumns(x => x.AiCrawledDetails)
  3716. .ExecuteCommandAsync();
  3717. });
  3718. return Ok(result.IsSuccess
  3719. ? JsonView(true, "操作成功")
  3720. : JsonView(false, $"操作失败:{result.ErrorMessage}"));
  3721. }
  3722. /// <summary>
  3723. /// 商邀资料AI-无团组版 混元AI查询资料(SSE流式推送)
  3724. /// </summary>
  3725. [HttpPost]
  3726. public async Task InvitationAI_NoGroupSearchStreamProgress([FromBody] InvitationAI_NoGroupSearchDto dto)
  3727. {
  3728. HttpContext.InitializeSse();
  3729. try
  3730. {
  3731. await HttpContext.SendSseStepAsync(5, "正在加载配置...");
  3732. #region 1. 异步并行化验证 (Performance Boost)
  3733. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI_NoGroup>().Where(x => x.IsDel == 0 && x.Id == dto.Id).FirstAsync();
  3734. var operatorName = await _sqlSugar.Queryable<Sys_Users>()
  3735. .Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId)
  3736. .Select(x => x.CnName).FirstAsync();
  3737. if (invAiInfo?.EntryInfo == null)
  3738. {
  3739. await HttpContext.SendSseStepAsync(-1, "未找到有效的关键字配置。");
  3740. return;
  3741. }
  3742. var entryInfo = invAiInfo.EntryInfo;
  3743. var targetCountrySet = new HashSet<string>(entryInfo.TargetCountry);
  3744. #endregion
  3745. await HttpContext.SendSseStepAsync(20, "正在解析本地典籍 (并行解密)...");
  3746. #region 2. 内存计算优化
  3747. // 仅查询必要字段,减少 DataReader 压力
  3748. var rawLocalDatas = await _sqlSugar.Queryable<Res_InvitationOfficialActivityData>()
  3749. .Where(x => x.IsDel == 0 && !string.IsNullOrEmpty(x.UnitName))
  3750. .Select(x => new { x.Country, x.UnitName, x.Address, x.Field, x.Contact, x.Tel, x.Email })
  3751. .ToListAsync();
  3752. // PLINQ 核心炼金:利用多核并行解密
  3753. var allDecrypted = rawLocalDatas.AsParallel().Select(item => new InvitationAI_NoGroupInfo
  3754. {
  3755. Guid = Guid.NewGuid().ToString("N"),
  3756. Source = 0, // 标识来源:本地
  3757. Region = AesEncryptionHelper.Decrypt(item.Country),
  3758. NameCn = AesEncryptionHelper.Decrypt(item.UnitName),
  3759. Address = AesEncryptionHelper.Decrypt(item.Address),
  3760. Scope = AesEncryptionHelper.Decrypt(item.Field),
  3761. Contact = AesEncryptionHelper.Decrypt(item.Contact),
  3762. Phone = AesEncryptionHelper.Decrypt(item.Tel),
  3763. Email = AesEncryptionHelper.Decrypt(item.Email),
  3764. OperatedAt = DateTime.Now,
  3765. Operator = operatorName,
  3766. }).ToList();
  3767. // 筛选符合国家的本地数据
  3768. var matchedCountries = allDecrypted.Where(x => targetCountrySet.Contains(x.Region)).ToList();
  3769. #endregion
  3770. #region 3. 混元 AI 协同炼金 (双阶段)
  3771. // --- 阶段 A: 行业匹配分析 ---
  3772. if (matchedCountries.Any())
  3773. {
  3774. await HttpContext.SendSseStepAsync(40, $"AI 正在执行 {entryInfo.TargetCountry.Count} 国的行业契合度分析...");
  3775. string industryQuestion = BuildIndustryPrompt(entryInfo, matchedCountries);
  3776. string industryRaw = await _hunyuanService.ChatCompletionsHunyuan_t1_latestAsync(industryQuestion);
  3777. var industryMatches = CleanAndParseJson<List<IndustryMatchResult>>(industryRaw) ?? new();
  3778. var matchedNames = new HashSet<string>(industryMatches.Select(x => x.TargetUnitName));
  3779. // 重新过滤:仅保留 AI 认为匹配的本地数据
  3780. matchedCountries = matchedCountries.Where(x => !string.IsNullOrEmpty(x.NameCn) && matchedNames.Contains(x.NameCn)).Distinct().ToList();
  3781. }
  3782. // --- 阶段 B: 差额补全 (Gap Filling) ---
  3783. var localInvDatas = new List<InvitationAI_NoGroupInfo>();
  3784. var aiTasks = new List<CountryAIPormptInfo>();
  3785. foreach (var country in entryInfo.TargetCountry)
  3786. {
  3787. var countryData = matchedCountries.Where(x => x.Region == country).Take(entryInfo.NeedCount).ToList();
  3788. localInvDatas.AddRange(countryData);
  3789. int gap = entryInfo.NeedCount - countryData.Count;
  3790. if (gap > 0) aiTasks.Add(new() { Country = country, Count = gap });
  3791. }
  3792. var hunyuanAIInvDatas = new List<InvitationAI_NoGroupInfo>();
  3793. if (aiTasks.Any())
  3794. {
  3795. // 强制冷却(应对 QPS 限制)
  3796. // 混元免费版或低阶版本通常有 1s/1次 的频率限制
  3797. await Task.Delay(1000);
  3798. // 任务配置计算
  3799. var countryTasks = QuotaScheduler.GenerateTasks(aiTasks, entryInfo.Industries, entryInfo.ScaleTypes);
  3800. var countryTasksGroupBy = countryTasks.GroupBy(x => x.Region).Select(g => new
  3801. {
  3802. Region = g.Key,
  3803. Counrt = g.Count()
  3804. }).ToList();
  3805. await HttpContext.SendSseStepAsync(60, $"AI 正在跨境检索 {string.Join(", ", countryTasksGroupBy.Select(x => $"{x.Region}({x.Counrt}条)"))} 单位资料...");
  3806. string searchQuestion = BuildHunyuanPrompt(aiTasks, countryTasks, entryInfo);
  3807. _logger.LogInformation(@"公务名称:{InvName}; 混元AI查询提示词:{searchQuestion}", invAiInfo.InvName, searchQuestion);
  3808. string searchRaw = await _hunyuanService.ChatCompletionsHunyuan_t1_latestAsync(searchQuestion);
  3809. var audit = ParseHunyuanResult<InvitationAI_NoGroupInfo>(searchRaw);
  3810. if (!audit.IsSuccess)
  3811. {
  3812. _logger.LogWarning(
  3813. "公务名称:{InvName};AI 检索失败;Step={Step};Reason={Reason}",
  3814. invAiInfo.InvName,
  3815. audit.AuditStep,
  3816. audit.AuditReason
  3817. );
  3818. await HttpContext.SendSseStepAsync(-1,
  3819. $"{GetFrontendAuditTitle(audit.AuditStep!)}:" +
  3820. $"{GetFrontendAuditReason(audit.AuditReason!)}");
  3821. return;
  3822. }
  3823. hunyuanAIInvDatas = audit.Data!
  3824. .Select(x =>
  3825. {
  3826. x.Guid = Guid.NewGuid().ToString("N");
  3827. x.Source = 1;
  3828. x.Operator = operatorName;
  3829. x.OperatedAt = DateTime.Now;
  3830. return x;
  3831. })
  3832. .ToList();
  3833. }
  3834. #endregion
  3835. await HttpContext.SendSseStepAsync(90, "正在同步成果至持久化仓库...");
  3836. #region 4. 数据融合与更新
  3837. var finalResult = localInvDatas.Concat(hunyuanAIInvDatas).ToList();
  3838. // 仅更新指定列,性能更优
  3839. invAiInfo.AiCrawledDetails = finalResult.OrderByDescending(x => x.OperatedAt).ToList();
  3840. bool isOk = await _sqlSugar.Updateable(invAiInfo)
  3841. .UpdateColumns(x => x.AiCrawledDetails)
  3842. .ExecuteCommandHasChangeAsync();
  3843. if (!isOk) await HttpContext.SendSseStepAsync(-1, $"数据持久化失败。");
  3844. #endregion
  3845. // 5. 终焉推送
  3846. if (!finalResult.Any())
  3847. {
  3848. await HttpContext.SendSseStepAsync(-1, "本次未检索到符合条件的境外单位,建议调整国家或行业范围。");
  3849. return;
  3850. }
  3851. await HttpContext.SendSseStepAsync(100,
  3852. "操作成功!资料已全部就绪。",
  3853. new
  3854. {
  3855. invAiInfo.Id,
  3856. AiCrawledDetails = finalResult
  3857. .OrderByDescending(x => x.Source)
  3858. .ThenBy(x => x.Region)
  3859. });
  3860. }
  3861. catch (Exception ex)
  3862. {
  3863. _logger.LogError(ex, "SSE 管道熔断");
  3864. await HttpContext.SendSseStepAsync(-1, $"SSE 错误:{ex.InnerException.Message}({ex.Message})");
  3865. }
  3866. finally
  3867. {
  3868. await HttpContext.FinalizeSseAsync();
  3869. }
  3870. }
  3871. /// <summary>
  3872. /// 商邀资料AI-无团组版 混元AI续写(SSE流式推送)
  3873. /// </summary>
  3874. /// <param name="dto"></param>
  3875. /// <returns></returns>
  3876. [HttpPost]
  3877. public async Task InvitationAI_NoGroupCompleteTextStream([FromBody] InvitationAI_NoGroupCompleteTextDto dto)
  3878. {
  3879. HttpContext.InitializeSse();
  3880. try
  3881. {
  3882. // 初始化检查 (Progress: 5%)
  3883. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI_NoGroup>().Where(x => x.Id == dto.Id).FirstAsync();
  3884. if (invAiInfo?.EntryInfo == null)
  3885. {
  3886. await HttpContext.SendSseStepAsync(-1, "请先设置关键字信息!");
  3887. return;
  3888. }
  3889. await HttpContext.SendSseStepAsync(10, "任务初始化完成,正在计算补齐缺口...");
  3890. // 任务拆解逻辑
  3891. var entryInfo = invAiInfo.EntryInfo;
  3892. var aiTasks = new List<CountryAIPormptInfo>();
  3893. int targetPerCountry = entryInfo.NeedCount;
  3894. foreach (var countryName in entryInfo.TargetCountry)
  3895. {
  3896. var countryDataCount = invAiInfo.AiCrawledDetails.Count(x => x.Region == countryName);
  3897. int aiNeedCount = targetPerCountry - countryDataCount;
  3898. if (aiNeedCount > 0) aiTasks.Add(new() { Country = countryName, Count = aiNeedCount });
  3899. }
  3900. if (!aiTasks.Any())
  3901. {
  3902. await HttpContext.SendSseStepAsync(100, "数据已满额,无需续写", invAiInfo.AiCrawledDetails);
  3903. return;
  3904. }
  3905. // 准备 AI Prompt (Progress: 20%)
  3906. var existingNames = new HashSet<string>(invAiInfo.AiCrawledDetails.Where(x => x.Source == 1).Select(x => x.NameCn));
  3907. string promptOther = $"请基于以下已存在的名称列表进行推荐,避免重复:{string.Join(", ", existingNames)}。{entryInfo.OtherConstraints}";
  3908. entryInfo.OtherConstraints = promptOther; // 将去重提示注入 entryInfo,确保 Prompt 构建时包含该信息
  3909. // 构建 Question
  3910. // 任务配置计算
  3911. var countryTasks = QuotaScheduler.GenerateTasks(aiTasks, entryInfo.Industries, entryInfo.ScaleTypes);
  3912. string question = BuildHunyuanPrompt(aiTasks, countryTasks, entryInfo);
  3913. await HttpContext.SendSseStepAsync(30, "AI 正在深度检索跨境商邀数据,请稍候...");
  3914. // 调用 AI (Progress: 30% - 80%)
  3915. string aiRawResponse = await _hunyuanService.ChatCompletionsHunyuan_t1_latestAsync(question);
  3916. await HttpContext.SendSseStepAsync(85, "数据已捕获,正在进行格式校验与去重...");
  3917. // 5. 解析与清洗数据
  3918. string operatorName = await _sqlSugar.Queryable<Sys_Users>().Where(x => x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  3919. var audit = ParseHunyuanResult<InvitationAI_NoGroupInfo>(aiRawResponse);
  3920. if (!audit.IsSuccess)
  3921. {
  3922. await HttpContext.SendSseStepAsync(-1, $"AI 续写失败:{GetFrontendAuditReason(audit.AuditReason!)}");
  3923. return;
  3924. }
  3925. if (audit.Data == null || !audit.Data.Any())
  3926. {
  3927. await HttpContext.SendSseStepAsync(-1, "AI 未生成任何新单位,建议调整国家或行业范围。");
  3928. return;
  3929. }
  3930. var hunyuanAIInvDatas = audit.Data
  3931. .Select(x =>
  3932. {
  3933. x.Guid = Guid.NewGuid().ToString("N");
  3934. x.Source = 1;
  3935. x.Operator = operatorName;
  3936. x.OperatedAt = DateTime.Now;
  3937. return x;
  3938. })
  3939. .ToList();
  3940. // 6. 数据库操作 (Progress: 95%)
  3941. invAiInfo.AiCrawledDetails.AddRange(hunyuanAIInvDatas);
  3942. invAiInfo.AiCrawledDetails = invAiInfo.AiCrawledDetails.OrderByDescending(x => x.OperatedAt).ToList();
  3943. var update = await _sqlSugar.Updateable(invAiInfo).UpdateColumns(x => x.AiCrawledDetails).ExecuteCommandAsync();
  3944. if (update > 0)
  3945. {
  3946. await HttpContext.SendSseStepAsync(100,
  3947. $"AI 续写成功!新增 {hunyuanAIInvDatas.Count} 条数据",
  3948. invAiInfo.AiCrawledDetails);
  3949. }
  3950. else
  3951. {
  3952. await HttpContext.SendSseStepAsync(-1, "数据库更新失败");
  3953. }
  3954. }
  3955. catch (Exception ex)
  3956. {
  3957. _logger.LogError(ex, "SSE 续写异常");
  3958. await HttpContext.SendSseStepAsync(-1, $"炼金炸炉:{ex.Message}");
  3959. }
  3960. finally
  3961. {
  3962. await HttpContext.FinalizeSseAsync();
  3963. }
  3964. }
  3965. /// <summary>
  3966. /// 商邀资料AI-无团组版
  3967. /// 文件生成(基于混元AI已爬取数据进行格式化输出,供用户下载使用)
  3968. /// </summary>
  3969. /// <returns></returns>
  3970. [HttpGet("{id}")]
  3971. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  3972. public async Task<IActionResult> InvitationAI_NoGroupFileGenerator(int id)
  3973. {
  3974. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI_NoGroup>()
  3975. .Where(x => x.Id == id)
  3976. .FirstAsync();
  3977. var dataList = invAiInfo?.AiCrawledDetails;
  3978. if (dataList == null)
  3979. return Ok(JsonView(false, "数据为空"));
  3980. Document doc = new Document();
  3981. DocumentBuilder builder = new DocumentBuilder(doc);
  3982. // A4页面设置
  3983. double usableWidth = 495;
  3984. foreach (Aspose.Words.Section sec in doc.Sections)
  3985. {
  3986. sec.PageSetup.PaperSize = Aspose.Words.PaperSize.A4;
  3987. sec.PageSetup.LeftMargin = 50;
  3988. sec.PageSetup.RightMargin = 50;
  3989. sec.PageSetup.TopMargin = 50;
  3990. sec.PageSetup.BottomMargin = 50;
  3991. }
  3992. // 卡片背景
  3993. Color cardBg = Color.FromArgb(252, 252, 252);
  3994. string fontName = "微软雅黑";
  3995. int count = 1;
  3996. foreach (var item in dataList)
  3997. {
  3998. // 标题
  3999. builder.ParagraphFormat.ClearFormatting();
  4000. builder.Font.ClearFormatting();
  4001. builder.Font.Name = fontName;
  4002. builder.Font.Size = 16;
  4003. builder.Font.Bold = true;
  4004. builder.Font.Color = Color.Black;
  4005. builder.Write($"{count++}. {item.NameCn}");
  4006. builder.InsertParagraph();
  4007. // 英文名
  4008. builder.Font.Size = 10;
  4009. builder.Font.Bold = false;
  4010. builder.Font.Color = Color.FromArgb(120, 120, 120);
  4011. builder.Write(item.NameEn ?? "");
  4012. builder.InsertParagraph();
  4013. builder.InsertParagraph();
  4014. // ====== 卡片表格 ======
  4015. Aspose.Words.Tables.Table table = builder.StartTable();
  4016. builder.CellFormat.Shading.BackgroundPatternColor = cardBg;
  4017. builder.CellFormat.VerticalAlignment = CellVerticalAlignment.Center;
  4018. builder.CellFormat.WrapText = true;
  4019. builder.CellFormat.LeftPadding = 12;
  4020. builder.CellFormat.RightPadding = 12;
  4021. builder.CellFormat.TopPadding = 8;
  4022. builder.CellFormat.BottomPadding = 8;
  4023. double labelW = 120;
  4024. AddSingleRow(builder, fontName, labelW, usableWidth - labelW, "详细地址", item.Address);
  4025. AddSingleRow(builder, fontName, labelW, usableWidth - labelW, "联系人", item.Contact);
  4026. AddSingleRow(builder, fontName, labelW, usableWidth - labelW, "联系电话", item.Phone);
  4027. AddSingleRow(builder, fontName, labelW, usableWidth - labelW, "电子邮箱", item.Email);
  4028. AddSingleLinkRow(builder, fontName, labelW, usableWidth - labelW, "官方网站", item.SiteUrl);
  4029. AddSingleLinkRow(builder, fontName, labelW, usableWidth - labelW, "最近三年动态", string.Join(", ", item.PostUrl.Select(p => p.Url)));
  4030. AddFullWidthRow(builder, fontName, labelW, usableWidth - labelW, "推荐等级", item.RecLevel);
  4031. AddFullWidthRow(builder, fontName, labelW, usableWidth - labelW, "经营范围", item.Scope);
  4032. AddFullWidthRow(builder, fontName, labelW, usableWidth - labelW, "对接建议", item.IntgAdvice);
  4033. // 表格样式
  4034. table.SetBorders(Aspose.Words.LineStyle.Single, 0.5, Color.FromArgb(230, 230, 230));
  4035. table.AutoFit(AutoFitBehavior.FixedColumnWidths);
  4036. table.AllowAutoFit = false;
  4037. table.PreferredWidth = PreferredWidth.FromPoints(usableWidth);
  4038. builder.EndTable();
  4039. builder.InsertParagraph();
  4040. builder.InsertParagraph();
  4041. }
  4042. // 保存逻辑
  4043. string fileName = $"{invAiInfo.InvName}(无团组版)_Professional.docx";
  4044. string filePath = Path.Combine(AppSettingsHelper.Get("InvitationAIAssistBasePath"), fileName);
  4045. doc.Save(filePath, Aspose.Words.SaveFormat.Docx);
  4046. return Ok(JsonView(true, "生成成功", new { Url = $"{AppSettingsHelper.Get("WordBaseUrl")}/{AppSettingsHelper.Get("InvitationAIAssistFtpPath")}/{fileName}" }));
  4047. }
  4048. /// <summary>
  4049. /// 商邀资料AI-无团组版
  4050. /// 生成邮件(SSE流式推送)
  4051. /// </summary>
  4052. /// <returns></returns>
  4053. [HttpPost]
  4054. public async Task InvitationAI_NoGroupGenerateEmailStream([FromBody] InvitationAI_NoGroupGenerateEmailDto dto)
  4055. {
  4056. HttpContext.InitializeSse();
  4057. try
  4058. {
  4059. // 1. 基础校验与资源加载 (Progress: 5%)
  4060. if (dto.Id < 1 || dto.Guids == null || !dto.Guids.Any())
  4061. {
  4062. await HttpContext.SendSseStepAsync(-1, "请求参数不完整,请选择单位");
  4063. return;
  4064. }
  4065. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI_NoGroup>().Where(x => x.Id == dto.Id).FirstAsync();
  4066. if (invAiInfo?.AiCrawledDetails == null)
  4067. {
  4068. await HttpContext.SendSseStepAsync(-1, "基础商邀资料不存在");
  4069. return;
  4070. }
  4071. await HttpContext.SendSseStepAsync(10, "AI 正在分析考察团组背景与访问意图...");
  4072. // 2. 准备 AI 上下文数据 (Progress: 15%)
  4073. var clientInfoSources = invAiInfo.AiCrawledDetails.Where(x => dto.Guids.Contains(x.Guid)).ToList();
  4074. var clientInfosForAI = clientInfoSources.Select(x => new AICreateEmailInfo()
  4075. {
  4076. Guid = x.Guid,
  4077. NameCn = x.NameCn,
  4078. Scope = x.Scope
  4079. }).ToList();
  4080. string operatorName = await _sqlSugar.Queryable<Sys_Users>()
  4081. .Where(x => x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  4082. // 3. 构建 Prompt 并调用 AI (Progress: 25%)
  4083. await HttpContext.SendSseStepAsync(30, $"AI 正在为 {clientInfosForAI.Count} 家单位撰写定制化正式邮件...");
  4084. // 3.1 构建 Prompt
  4085. string pormpt = BuildHunyuanEmailPrompt(
  4086. invAiInfo.EntryInfo?.OriginUnit ?? "",
  4087. invAiInfo.EntryInfo?.Objective ?? "",
  4088. invAiInfo.EntryInfo?.VisitDate ?? "",
  4089. clientInfosForAI
  4090. );
  4091. // 调用 AI (此处为阻塞式等待 AI 结果,若混元支持 Stream 可进一步拆解)
  4092. string aiResponse = await _hunyuanService.ChatCompletionsHunyuan_t1_latestAsync(pormpt);
  4093. await HttpContext.SendSseStepAsync(80, "邮件初稿已生成,正在进行 HTML 格式校验与转义处理...");
  4094. // 4. 解析结果 (Progress: 85%)
  4095. var hunyuanAIEmailDatas = new List<AICreateEmailInfo>();
  4096. if (!string.IsNullOrWhiteSpace(aiResponse))
  4097. {
  4098. // 预处理:过滤 AI 可能返回的 Markdown 标记
  4099. string cleanJson = aiResponse.Trim();
  4100. if (cleanJson.StartsWith("```json"))
  4101. cleanJson = cleanJson.Substring(7, cleanJson.Length - 10).Trim();
  4102. else if (cleanJson.StartsWith("```"))
  4103. cleanJson = cleanJson.Substring(3, cleanJson.Length - 6).Trim();
  4104. try
  4105. {
  4106. // 解析并注入 Source 标识
  4107. hunyuanAIEmailDatas = JsonConvert.DeserializeObject<List<AICreateEmailInfo>>(cleanJson);
  4108. }
  4109. catch (JsonException ex)
  4110. {
  4111. // 记录日志并考虑 fallback 策略
  4112. _logger.LogError(ex, "Hunyuan AI 响应解析失败。原始数据:{Response}", aiResponse);
  4113. }
  4114. }
  4115. if (hunyuanAIEmailDatas == null || !hunyuanAIEmailDatas.Any())
  4116. {
  4117. await HttpContext.SendSseStepAsync(-1, "AI 格式解析失败,请尝试重新生成");
  4118. return;
  4119. }
  4120. // 5. 更新数据模型 (Progress: 90%)
  4121. foreach (var client in clientInfoSources)
  4122. {
  4123. var aiEmail = hunyuanAIEmailDatas.FirstOrDefault(x => x.Guid == client.Guid);
  4124. if (aiEmail != null)
  4125. {
  4126. client.EmailInfo.Status = 2; // 已生成
  4127. client.EmailInfo.EmailTitle = aiEmail.Subject;
  4128. client.EmailInfo.EmailContent = aiEmail.Content;
  4129. client.EmailInfo.Operator = operatorName;
  4130. client.EmailInfo.OperatedAt = DateTime.Now;
  4131. }
  4132. }
  4133. // 6. 数据库持久化 (Progress: 95%)
  4134. // 采用增量更新策略,避免直接 Where 过滤掉其他未选中的数据
  4135. var update = await _sqlSugar.Updateable(invAiInfo).UpdateColumns(x => x.AiCrawledDetails).ExecuteCommandAsync();
  4136. if (update > 0)
  4137. {
  4138. await HttpContext.SendSseStepAsync(100, "邮件全部生成完毕并已存入团组资料库", invAiInfo.AiCrawledDetails);
  4139. }
  4140. else
  4141. {
  4142. await HttpContext.SendSseStepAsync(-1, "数据库写入失败,请联系管理员");
  4143. }
  4144. }
  4145. catch (Exception ex)
  4146. {
  4147. _logger.LogError(ex, "邮件生成异常");
  4148. await HttpContext.SendSseStepAsync(-1, $"生成失败:{ex.Message}");
  4149. }
  4150. finally
  4151. {
  4152. await HttpContext.FinalizeSseAsync();
  4153. }
  4154. }
  4155. /// <summary>
  4156. /// 商邀资料AI-无团组版
  4157. /// 邮件保存
  4158. /// </summary>
  4159. /// <returns></returns>
  4160. [HttpPost]
  4161. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  4162. public async Task<IActionResult> InvitationAI_NoGroupEmailSave(InvitationAI_NoGroupEmailSaveDto dto)
  4163. {
  4164. if (dto.Id < 1) return Ok(JsonView(false, "请选择保存的邮件资料"));
  4165. if (dto.CurrUserId < 1) return Ok(JsonView(false, "请传入用户Id"));
  4166. if (string.IsNullOrEmpty(dto.Guid)) return Ok(JsonView(false, "请传入Guid"));
  4167. if (string.IsNullOrEmpty(dto.EmailTitle)) return Ok(JsonView(false, "请传入邮件标题"));
  4168. if (string.IsNullOrEmpty(dto.EmailContent)) return Ok(JsonView(false, "请传入邮件内容"));
  4169. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI_NoGroup>()
  4170. .Where(x => x.Id == dto.Id)
  4171. .FirstAsync();
  4172. if (invAiInfo == null) return Ok(JsonView(false, "数据信息为空"));
  4173. var dataList = invAiInfo?.AiCrawledDetails;
  4174. if (dataList == null) return Ok(JsonView(false, "邀请方数据信息为空"));
  4175. var editInfo = dataList.Where(x => x.Guid == dto.Guid).FirstOrDefault();
  4176. if (editInfo == null) return Ok(JsonView(false, "邀请方信息为空"));
  4177. var opUserName = await _sqlSugar.Queryable<Sys_Users>().Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  4178. editInfo.EmailInfo.EmailTitle = dto.EmailTitle;
  4179. editInfo.EmailInfo.EmailContent = dto.EmailContent;
  4180. editInfo.EmailInfo.Operator = opUserName;
  4181. editInfo.EmailInfo.OperatedAt = DateTime.Now;
  4182. var datas = dataList.Where(x => x.Guid != dto.Guid).ToList();
  4183. datas.Add(editInfo);
  4184. // 排序
  4185. invAiInfo.AiCrawledDetails = datas.OrderByDescending(x => x.OperatedAt).ToList();
  4186. var editUpd = await _sqlSugar.Updateable(invAiInfo).UpdateColumns(x => x.AiCrawledDetails).ExecuteCommandAsync();
  4187. if (editUpd < 1)
  4188. {
  4189. return Ok(JsonView(true, "邮件保存失败"));
  4190. }
  4191. return Ok(JsonView(true, "邮件保存成功"));
  4192. }
  4193. /// <summary>
  4194. /// 商邀资料AI-无团组版 邮箱附件上传
  4195. /// </summary>
  4196. [HttpPost]
  4197. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  4198. public async Task<IActionResult> InvitationAI_NoGroupFileSave([FromForm] InvitationAI_NoGroupFileSaveDto dto)
  4199. {
  4200. // 1. 炼金前置:严格参数校验
  4201. if (dto.Id < 1) return Ok(JsonView(false, "请选择保存的公务资料"));
  4202. if (dto.CurrUserId < 1) return Ok(JsonView(false, "请传入用户Id"));
  4203. if (string.IsNullOrEmpty(dto.Guid)) return Ok(JsonView(false, "请传入Guid"));
  4204. if (dto.Attachments == null || !dto.Attachments.Any()) return Ok(JsonView(false, "请传入附件"));
  4205. // 2. 数据获取:利用 SqlSugar 异步查询
  4206. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI_NoGroup>()
  4207. .InSingleAsync(dto.Id);
  4208. if (invAiInfo?.AiCrawledDetails == null) return Ok(JsonView(false, "邀请方信息不存在"));
  4209. // 3. 定位到具体需要编辑的 Guid 记录
  4210. var editInfo = invAiInfo.AiCrawledDetails.FirstOrDefault(x => x.Guid == dto.Guid);
  4211. if (editInfo == null) return Ok(JsonView(false, "未找到匹配的 Guid 记录"));
  4212. // 4. 追踪信息更新
  4213. var opUserName = await _sqlSugar.Queryable<Sys_Users>()
  4214. .Where(x => x.Id == dto.CurrUserId && x.IsDel == 0)
  4215. .Select(x => x.CnName)
  4216. .FirstAsync() ?? "-";
  4217. // 更新操作人与时间 (利用引用类型特性)
  4218. editInfo.Operator = opUserName;
  4219. editInfo.OperatedAt = DateTime.Now;
  4220. // 初始化 EmailInfo 确保不为 null
  4221. editInfo.EmailInfo ??= new EmailInfo();
  4222. editInfo.EmailInfo.Operator = opUserName;
  4223. editInfo.EmailInfo.OperatedAt = DateTime.Now;
  4224. editInfo.EmailInfo.AttachmentPaths ??= new List<string>();
  4225. // 5. 构建绝对路径与相对路径
  4226. string dirName = $"{editInfo.NameEn?.Trim() ?? "Default"}_{dto.Guid}";
  4227. string baseDir = AppSettingsHelper.Get("InvitationAIAssistBasePath");
  4228. string ftpBase = AppSettingsHelper.Get("InvitationAIAssistFtpPath");
  4229. string absolutePath = Path.Combine(baseDir, dirName);
  4230. string relativePathPrefix = $"{ftpBase}{dirName}/";
  4231. // 核心修复:确保父级目录递归创建,防止 DirectoryNotFoundException
  4232. if (!Directory.Exists(absolutePath))
  4233. {
  4234. Directory.CreateDirectory(absolutePath);
  4235. }
  4236. var newSavedRelativePaths = new List<string>();
  4237. // 6. 持续文件存储与实时验证
  4238. foreach (var file in dto.Attachments)
  4239. {
  4240. // 文件名安全过滤
  4241. string safeName = string.Join("_", file.FileName.Split(Path.GetInvalidFileNameChars()));
  4242. string fullPath = Path.Combine(absolutePath, safeName);
  4243. // 重名冲突处理:文件名(n).ext
  4244. if (System.IO.File.Exists(fullPath))
  4245. {
  4246. string fileNameOnly = Path.GetFileNameWithoutExtension(safeName);
  4247. string extension = Path.GetExtension(safeName);
  4248. int count = 1;
  4249. while (System.IO.File.Exists(fullPath))
  4250. {
  4251. safeName = $"{fileNameOnly}({count++}){extension}";
  4252. fullPath = Path.Combine(absolutePath, safeName);
  4253. }
  4254. }
  4255. try
  4256. {
  4257. // 写入流:使用作用域隔离,确保退出 using 时立即释放句柄
  4258. using (var stream = new FileStream(fullPath, FileMode.Create, FileAccess.Write, FileShare.None))
  4259. {
  4260. await file.CopyToAsync(stream);
  4261. await stream.FlushAsync(); // 强制落盘
  4262. }
  4263. // 实时验证:物理验证 + 逻辑校验
  4264. var fileInfo = new FileInfo(fullPath);
  4265. if (fileInfo.Exists && fileInfo.Length == file.Length)
  4266. {
  4267. newSavedRelativePaths.Add($"{relativePathPrefix}{safeName}");
  4268. }
  4269. }
  4270. catch (Exception ex)
  4271. {
  4272. // 发生 IO 异常时清理残余文件并抛出,触发外层处理
  4273. if (System.IO.File.Exists(fullPath)) System.IO.File.Delete(fullPath);
  4274. // 此处记录日志:
  4275. _logger.LogError(ex, "File Save Error");
  4276. continue; // 跳过失败的文件,继续下一个
  4277. }
  4278. }
  4279. if (!newSavedRelativePaths.Any()) return Ok(JsonView(false, "所有文件上传均失败"));
  4280. // 7. 数据合并与持久化:
  4281. // 将新路径与旧路径合并,并去重
  4282. var updatedPaths = editInfo.EmailInfo.AttachmentPaths;
  4283. updatedPaths.AddRange(newSavedRelativePaths);
  4284. editInfo.EmailInfo.AttachmentPaths = updatedPaths.Distinct().ToList();
  4285. // 重新排序整体列表
  4286. invAiInfo.AiCrawledDetails = invAiInfo.AiCrawledDetails
  4287. .OrderByDescending(x => x.OperatedAt)
  4288. .ToList();
  4289. // SqlSugar 高效部分列更新
  4290. var isSuccess = await _sqlSugar.Updateable(invAiInfo)
  4291. .UpdateColumns(x => x.AiCrawledDetails)
  4292. .ExecuteCommandAsync() > 0;
  4293. if (!isSuccess) return Ok(JsonView(false, "数据库更新失败"));
  4294. // 8. 返回前端数据 (支持 lowerCamelCase)
  4295. var officeBaseUrl = AppSettingsHelper.Get("OfficeBaseUrl");
  4296. var finalResultUrls = editInfo.EmailInfo.AttachmentPaths
  4297. .Select(path => $"{officeBaseUrl}{path}")
  4298. .ToList();
  4299. return Ok(JsonView(true, "邮件附件保存成功", finalResultUrls));
  4300. }
  4301. /// <summary>
  4302. /// 商邀资料AI-无团组版 邮箱附件删除
  4303. /// </summary>
  4304. [HttpPost]
  4305. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  4306. public async Task<IActionResult> InvitationAI_NoGroupFileDel([FromBody] InvitationAI_NoGroupFileDelDto dto)
  4307. {
  4308. // 1. 基础校验
  4309. if (dto.Id < 1) return Ok(JsonView(false, "请选择要删除的邮件"));
  4310. if (string.IsNullOrEmpty(dto.Guid)) return Ok(JsonView(false, "请传入Guid"));
  4311. if (dto.AttachmentNames == null || !dto.AttachmentNames.Any()) return Ok(JsonView(false, "请传入待删除附件名称"));
  4312. // 2. 数据获取
  4313. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI_NoGroup>().InSingleAsync(dto.Id);
  4314. if (invAiInfo?.AiCrawledDetails == null) return Ok(JsonView(false, "业务数据不存在"));
  4315. var editInfo = invAiInfo.AiCrawledDetails.FirstOrDefault(x => x.Guid == dto.Guid);
  4316. if (editInfo?.EmailInfo?.AttachmentPaths == null) return Ok(JsonView(false, "未找到对应的附件记录"));
  4317. // 3. 操作人追踪
  4318. var opUserName = await _sqlSugar.Queryable<Sys_Users>()
  4319. .Where(x => x.Id == dto.CurrUserId && x.IsDel == 0)
  4320. .Select(x => x.CnName).FirstAsync() ?? "System";
  4321. // 更新追踪状态
  4322. editInfo.Operator = editInfo.EmailInfo.Operator = opUserName;
  4323. editInfo.OperatedAt = editInfo.EmailInfo.OperatedAt = DateTime.Now;
  4324. // 4. 路径
  4325. string dirName = $"{editInfo.NameEn?.Trim() ?? "Default"}_{dto.Guid}";
  4326. string absoluteDir = Path.Combine(AppSettingsHelper.Get("InvitationAIAssistBasePath"), dirName);
  4327. // 获取当前数据库中的相对路径列表
  4328. var currentPaths = editInfo.EmailInfo.AttachmentPaths;
  4329. bool isChanged = false;
  4330. // 5. 核心删除逻辑
  4331. foreach (var fileName in dto.AttachmentNames)
  4332. {
  4333. // 查找数据库中是否存在包含该文件名的路径 (忽略大小写比较)
  4334. var targetRelativePath = currentPaths.FirstOrDefault(p => p.EndsWith("/" + fileName) || p.Equals(fileName));
  4335. if (targetRelativePath != null)
  4336. {
  4337. // A. 从数据库记录中移除
  4338. currentPaths.Remove(targetRelativePath);
  4339. isChanged = true;
  4340. // B. 物理文件删除逻辑
  4341. // 注意:fullPath 必须是 [基础路径] + [目录名] + [纯文件名]
  4342. string fullPath = Path.Combine(absoluteDir, fileName);
  4343. try
  4344. {
  4345. if (System.IO.File.Exists(fullPath))
  4346. {
  4347. System.IO.File.Delete(fullPath);
  4348. }
  4349. }
  4350. catch (Exception ex)
  4351. {
  4352. // 记录 IO 异常但不中断流程
  4353. _logger.LogWarning($"物理文件删除失败: {fullPath}, {ex.Message}");
  4354. }
  4355. }
  4356. }
  4357. if (!isChanged) return Ok(JsonView(false, "未找到匹配的可删除附件"));
  4358. // 6. 数据同步与持久化
  4359. editInfo.EmailInfo.AttachmentPaths = currentPaths;
  4360. // 排序 (利用引用类型,无需手动 Add/Remove)
  4361. invAiInfo.AiCrawledDetails = invAiInfo.AiCrawledDetails.OrderByDescending(x => x.OperatedAt).ToList();
  4362. var isUpd = await _sqlSugar.Updateable(invAiInfo)
  4363. .UpdateColumns(x => x.AiCrawledDetails)
  4364. .ExecuteCommandAsync() > 0;
  4365. if (!isUpd) return Ok(JsonView(false, "数据库记录更新失败"));
  4366. // 7. 返回剩余附件的完整访问地址
  4367. var officeBaseUrl = AppSettingsHelper.Get("OfficeBaseUrl");
  4368. var remainingUrls = editInfo.EmailInfo.AttachmentPaths
  4369. .Select(path => $"{officeBaseUrl}{path}")
  4370. .ToList();
  4371. return Ok(JsonView(true, "附件删除成功", remainingUrls));
  4372. }
  4373. /// <summary>
  4374. /// 商邀资料AI-无团组版 发送邮件(SSE 流式推送)
  4375. /// </summary>
  4376. [HttpPost]
  4377. public async Task InvitationAI_NoGroupSeedEmailStream([FromBody] InvitationAI_NoGroupSeedEmailDto dto)
  4378. {
  4379. // 1. 初始化 SSE
  4380. HttpContext.InitializeSse();
  4381. try
  4382. {
  4383. await HttpContext.SendSseStepAsync(5, "正在准备发送邮件队列...");
  4384. #region 1. 参数与权限前置校验
  4385. if (dto.Id < 1 || dto.CurrUserId < 1 || dto.Guids == null || !dto.Guids.Any())
  4386. {
  4387. await HttpContext.SendSseStepAsync(-1, "参数验证失败,请检查选择的单位及用户状态。");
  4388. return;
  4389. }
  4390. // hotmail 配置信息验证
  4391. var hotmailConfig = await _hotmailService.GetUserMailConfig(dto.CurrUserId);
  4392. (bool verify, string msg) = _hotmailService.ConfigVerify(hotmailConfig);
  4393. if (!verify)
  4394. {
  4395. await HttpContext.SendSseStepAsync(-1, msg);
  4396. return;
  4397. }
  4398. // 获取商邀信息和用户信息
  4399. var invAiInfo = await _sqlSugar.Queryable<Res_InvitationAI_NoGroup>().FirstAsync(x => x.IsDel == 0 && x.Id == dto.Id);
  4400. var userInfo = await _sqlSugar.Queryable<Sys_Users>()
  4401. .Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId)
  4402. .Select(x => new { x.Email, x.CnName })
  4403. .FirstAsync();
  4404. if (invAiInfo?.AiCrawledDetails == null)
  4405. {
  4406. await HttpContext.SendSseStepAsync(-1, "未找到有效的邀请方数据。");
  4407. return;
  4408. }
  4409. // 提取待发送的目标集合
  4410. var guidSet = new HashSet<string>(dto.Guids);
  4411. var seedInvInfos = invAiInfo.AiCrawledDetails.Where(x => guidSet.Contains(x.Guid)).ToList();
  4412. if (!seedInvInfos.Any())
  4413. {
  4414. await HttpContext.SendSseStepAsync(-1, "所选单位信息在原始库中不存在。");
  4415. return;
  4416. }
  4417. #endregion
  4418. await HttpContext.SendSseStepAsync(15, $"准备就绪,共计 {seedInvInfos.Count} 封邮件待发送...");
  4419. #region 2. 批量发送逻辑 (流式反馈)
  4420. int total = seedInvInfos.Count;
  4421. int current = 0;
  4422. var successCount = 0;
  4423. var failCount = 0;
  4424. foreach (var item in seedInvInfos)
  4425. {
  4426. current++;
  4427. // 计算进度:从 20% 到 90%
  4428. int progress = 20 + (int)((double)current / total * 70);
  4429. if (string.IsNullOrEmpty(item.EmailInfo?.EmailTitle) || string.IsNullOrEmpty(item.EmailInfo?.EmailContent))
  4430. {
  4431. await HttpContext.SendSseStepAsync(progress, $"跳过:{item.NameCn} (邮件标题或内容缺失)");
  4432. failCount++;
  4433. continue;
  4434. }
  4435. try
  4436. {
  4437. var req = new MailDto()
  4438. {
  4439. Subject = item.EmailInfo.EmailTitle,
  4440. Content = item.EmailInfo.EmailContent,
  4441. To = item.Email,
  4442. AttachmentPaths = item.EmailInfo.AttachmentPaths
  4443. };
  4444. var res = await _hotmailService.SendMailAsync(hotmailConfig.UserName, req);
  4445. if (res.IsSuccess)
  4446. {
  4447. successCount++;
  4448. // 更新本地状态
  4449. item.EmailInfo.Status = 4; // 发送成功状态
  4450. item.EmailInfo.Operator = userInfo.CnName;
  4451. item.EmailInfo.OperatedAt = DateTime.Now;
  4452. await HttpContext.SendSseStepAsync(progress, $"成功:已向 {item.NameCn}({item.Email}) 发送邮件");
  4453. }
  4454. else
  4455. {
  4456. failCount++;
  4457. await HttpContext.SendSseStepAsync(progress, $"失败:{item.NameCn} 发送失败 ({res.Message})");
  4458. }
  4459. }
  4460. catch (Exception ex)
  4461. {
  4462. failCount++;
  4463. _logger.LogError(ex, $"企微邮件推送异常:{item.NameCn}");
  4464. await HttpContext.SendSseStepAsync(progress, $"异常:{item.NameCn} 连接超时");
  4465. }
  4466. }
  4467. #endregion
  4468. await HttpContext.SendSseStepAsync(95, "正在归档发送记录...");
  4469. #region 3. 数据同步与收尾
  4470. // 更新内存中的全量数据:排除旧的,加入已更新状态的
  4471. invAiInfo.AiCrawledDetails = invAiInfo.AiCrawledDetails
  4472. .Where(x => !guidSet.Contains(x.Guid))
  4473. .Concat(seedInvInfos)
  4474. .OrderByDescending(x => x.EmailInfo.OperatedAt)
  4475. .ToList();
  4476. // 精准更新 JSON 列
  4477. var updateOk = await _sqlSugar.Updateable(invAiInfo)
  4478. .UpdateColumns(x => x.AiCrawledDetails)
  4479. .ExecuteCommandHasChangeAsync();
  4480. if (!updateOk)
  4481. {
  4482. await HttpContext.SendSseStepAsync(-1, "记录保存失败,请检查数据库连接。");
  4483. return;
  4484. }
  4485. await HttpContext.SendSseStepAsync(100, $"任务完成!成功: {successCount}, 失败: {failCount}", new
  4486. {
  4487. Id = invAiInfo.Id,
  4488. SuccessCount = successCount,
  4489. FailCount = failCount
  4490. });
  4491. #endregion
  4492. }
  4493. catch (Exception ex)
  4494. {
  4495. _logger.LogError(ex, "邮件发送流发生崩溃");
  4496. await HttpContext.SendSseStepAsync(-1, $"系统错误:{ex.Message}");
  4497. }
  4498. finally
  4499. {
  4500. await HttpContext.FinalizeSseAsync();
  4501. }
  4502. }
  4503. #endregion
  4504. #region 公务出访
  4505. /// <summary>
  4506. /// 获取团组所有信息,绑定下拉框
  4507. /// </summary>
  4508. /// <param name="dto"></param>
  4509. /// <returns></returns>
  4510. [HttpPost]
  4511. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  4512. public async Task<IActionResult> GetGroupAllList(OfficialActivitiesByDiIdDto dto)
  4513. {
  4514. //string groupSql = string.Format("Select * From Grp_DelegationInfo With(NoLock) Where IsDel = 0 Order By CreateTime Desc");
  4515. //List<Grp_DelegationInfo> _DelegationInfos = _sqlSugar.SqlQueryable< Grp_DelegationInfo >(groupSql).ToList();
  4516. var _groupData = await _delegationInfoRep.PostShareGroupInfos(1);
  4517. dynamic groupInfos = null;
  4518. if (_groupData.Code == 0) groupInfos = _groupData.Data;
  4519. var data = _sqlSugar.Queryable<Sys_SetData>().Where(a => a.IsDel == 0).ToList();
  4520. var data1 = data.Where(a => a.STid == 38).Select(x => new { x.Id, x.Name }).ToList();
  4521. var data2 = data.Where(a => a.STid == 101).Select(x => new { x.Id, x.Name }).ToList();
  4522. //张总安排未参与对接
  4523. if (data2.Any(x => x.Name.Equals("张总安排未参与对接")))
  4524. {
  4525. var zhangZong = data2.FirstOrDefault(x => x.Name.Equals("张总安排未参与对接"));
  4526. if (zhangZong != null) data2.Remove(zhangZong);
  4527. data2.Insert(0, zhangZong);
  4528. }
  4529. var data3 = data.Where(a => a.STid == 66).Select(x => new { x.Id, x.Name, x.Remark }).ToList();
  4530. var _DeleFile = _sqlSugar.Queryable<Grp_DeleFile>().Where(a => a.Diid == dto.DiId && a.IsDel == 0 && a.Category == 970).ToList();
  4531. var translatorData = _sqlSugar.Queryable<Res_TranslatorLibrary>().Where(a => a.IsDel == 0).ToList();
  4532. var translatorData1 = _mapper.Map<List<TranslatorView>>(translatorData);
  4533. foreach (var item in translatorData1) EncryptionProcessor.DecryptProperties(item);
  4534. return Ok(JsonView(true,
  4535. "查询成功!",
  4536. new
  4537. {
  4538. Delegation = groupInfos,
  4539. SetData = data1,
  4540. DataSource = data2,
  4541. currencyData = data3,
  4542. DeleFile = _DeleFile,
  4543. TranslatorData = translatorData1
  4544. }));
  4545. }
  4546. /// <summary>
  4547. /// 公务 团组名称 Page List
  4548. /// </summary>
  4549. /// <param name="dto"></param>
  4550. /// <returns></returns>
  4551. [HttpPost]
  4552. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  4553. public async Task<IActionResult> OfficialActivitiesGroupNameList(OfficialActivitiesGroupNameListDto dto)
  4554. {
  4555. var sw = Stopwatch.StartNew();
  4556. if (!SharingStaticData.PortTypes.Contains(dto.PortType))
  4557. {
  4558. return Ok(JsonView(false, MsgTips.Port));
  4559. }
  4560. dto.PageIndex = dto.PageIndex == 0 ? 1 : dto.PageIndex;
  4561. dto.PageSize = dto.PageSize == 0 ? 10 : dto.PageSize;
  4562. RefAsync<int> total = 0;
  4563. // 只查询需要的字段,避免Select后再赋值,减少内存占用
  4564. var groupInfos = await _officialActivitiesRep._sqlSugar
  4565. .Queryable<Grp_DelegationInfo>()
  4566. .Where(x => x.IsDel == 0)
  4567. .WhereIF(!string.IsNullOrEmpty(dto.GroupName), x => x.TeamName.Contains(dto.GroupName))
  4568. .OrderByDescending(x => x.VisitDate)
  4569. .Select(x => new Web_ShareGroupInfoView()
  4570. {
  4571. Id = x.Id,
  4572. TeamName = x.TeamName,
  4573. TourCode = x.TourCode,
  4574. ClientName = x.ClientName,
  4575. VisitCountry = x.VisitCountry,
  4576. VisitStartDate = x.VisitStartDate,
  4577. VisitEndDate = x.VisitEndDate,
  4578. VisitDays = x.VisitDays,
  4579. VisitPNumber = x.VisitPNumber
  4580. })
  4581. .ToPageListAsync(dto.PageIndex, dto.PageSize, total);
  4582. if (groupInfos.Count > 0)
  4583. {
  4584. for (int i = 0; i < groupInfos.Count; i++)
  4585. {
  4586. var visitCountry = groupInfos[i].VisitCountry;
  4587. groupInfos[i].VisitCountry = !string.IsNullOrEmpty(visitCountry)
  4588. ? _delegationInfoRep.FormartTeamName(visitCountry)
  4589. : string.Empty;
  4590. }
  4591. }
  4592. sw.Stop();
  4593. return Ok(JsonView(true, $"操作成功!耗时:{sw.ElapsedMilliseconds}ms", groupInfos, total));
  4594. }
  4595. /// <summary>
  4596. /// 公务 翻译人员 Page List
  4597. /// </summary>
  4598. /// <param name="dto"></param>
  4599. /// <returns></returns>
  4600. [HttpPost]
  4601. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  4602. public async Task<IActionResult> OfficialActivitiesTranslatorList(OfficialActivitiesTranslatorListDto dto)
  4603. {
  4604. var sw = Stopwatch.StartNew();
  4605. if (!SharingStaticData.PortTypes.Contains(dto.PortType))
  4606. {
  4607. return Ok(JsonView(false, MsgTips.Port));
  4608. }
  4609. dto.PageIndex = dto.PageIndex == 0 ? 1 : dto.PageIndex;
  4610. dto.PageSize = dto.PageSize == 0 ? 10 : dto.PageSize;
  4611. var translatorIds = new List<int>();
  4612. if (!string.IsNullOrEmpty(dto.TranslatorName))
  4613. {
  4614. var translators = await _sqlSugar.Queryable<Res_TranslatorLibrary>()
  4615. .Where(x => x.IsDel == 0 && !string.IsNullOrEmpty(x.Name))
  4616. .Select(x => new { x.Id, x.Name })
  4617. .ToListAsync();
  4618. if (translators.Any())
  4619. {
  4620. foreach (var item in translators)
  4621. {
  4622. var decryptedName = AesEncryptionHelper.Decrypt(item.Name);
  4623. if (!string.IsNullOrEmpty(decryptedName) && decryptedName.Contains(dto.TranslatorName))
  4624. {
  4625. translatorIds.Add(item.Id);
  4626. }
  4627. }
  4628. }
  4629. if (translatorIds.Count == 0)
  4630. {
  4631. sw.Stop();
  4632. return Ok(JsonView(true, $"暂无翻译人员信息!耗时: {sw.ElapsedMilliseconds}ms", Array.Empty<TranslatorView>(), 0));
  4633. }
  4634. }
  4635. RefAsync<int> total = 0;
  4636. var translatorInfos = await _officialActivitiesRep._sqlSugar
  4637. .Queryable<Res_TranslatorLibrary>()
  4638. .Where(x => x.IsDel == 0 && !string.IsNullOrEmpty(x.Name))
  4639. .WhereIF(translatorIds != null && translatorIds.Count > 0, x => translatorIds.Contains(x.Id))
  4640. .OrderByDescending(x => x.CreateTime)
  4641. .Select(x => new TranslatorView()
  4642. {
  4643. Id = x.Id,
  4644. Area = x.Area,
  4645. Name = x.Name,
  4646. Sex = x.Sex,
  4647. Tel = x.Tel,
  4648. Email = x.Email,
  4649. WechatNo = x.WechatNo,
  4650. OtherSocialAccounts = x.OtherSocialAccounts,
  4651. Language = x.Language,
  4652. Price = x.Price,
  4653. Currency = x.Currency,
  4654. })
  4655. .ToPageListAsync(dto.PageIndex, dto.PageSize, total);
  4656. for (int i = 0; i < translatorInfos.Count; i++)
  4657. {
  4658. EncryptionProcessor.DecryptProperties(translatorInfos[i]);
  4659. }
  4660. sw.Stop();
  4661. string msg = translatorInfos.Count == 0 ? $"暂无您查询的翻译人员信息!耗时: {sw.ElapsedMilliseconds}ms" : $"操作成功!耗时: {sw.ElapsedMilliseconds}ms";
  4662. return Ok(JsonView(true, msg, translatorInfos, total));
  4663. }
  4664. /// <summary>
  4665. /// 公务 绑定下拉框
  4666. /// </summary>
  4667. /// <param name="dto"></param>
  4668. /// <returns></returns>
  4669. [HttpPost]
  4670. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  4671. public async Task<IActionResult> OfficialActivitiesInitDatabase(OfficialActivitiesByDiIdDto dto)
  4672. {
  4673. var data = _sqlSugar.Queryable<Sys_SetData>().Where(a => a.IsDel == 0).ToList();
  4674. var data1 = data.Where(a => a.STid == 38).Select(x => new { x.Id, x.Name }).ToList();
  4675. var data2 = data.Where(a => a.STid == 101).Select(x => new { x.Id, x.Name }).ToList();
  4676. //张总安排未参与对接
  4677. if (data2.Any(x => x.Name.Equals("张总安排未参与对接")))
  4678. {
  4679. var zhangZong = data2.FirstOrDefault(x => x.Name.Equals("张总安排未参与对接"));
  4680. if (zhangZong != null) data2.Remove(zhangZong);
  4681. data2.Insert(0, zhangZong);
  4682. }
  4683. var data3 = data.Where(a => a.STid == 66).Select(x => new { x.Id, x.Name, x.Remark }).ToList();
  4684. var _DeleFile = _sqlSugar.Queryable<Grp_DeleFile>().Where(a => a.Diid == dto.DiId && a.IsDel == 0 && a.Category == 970).ToList();
  4685. return Ok(JsonView(true,
  4686. "查询成功!",
  4687. new
  4688. {
  4689. SetData = data1,
  4690. DataSource = data2,
  4691. currencyData = data3,
  4692. DeleFile = _DeleFile
  4693. }));
  4694. }
  4695. /// <summary>
  4696. /// 公务List
  4697. /// </summary>
  4698. /// <param name="dto"></param>
  4699. /// <returns></returns>
  4700. [HttpPost]
  4701. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  4702. public async Task<IActionResult> QueryOfficialActivitiesByDiId(OfficialActivitiesByDiIdDto dto)
  4703. {
  4704. return Ok(await _officialActivitiesRep.QueryOfficialActivitiesByDiId(dto));
  4705. }
  4706. /// <summary>
  4707. /// 根据公务出访数据Id查询数据
  4708. /// </summary>
  4709. /// <param name="dto"></param>
  4710. /// <returns></returns>
  4711. [HttpPost]
  4712. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  4713. public async Task<IActionResult> QueryOfficialActivitiesById(OfficialActivitiesDiIdDto dto)
  4714. {
  4715. Result groupData = await _officialActivitiesRep.QueryOfficialActivitiesById(dto);
  4716. if (groupData.Code != 0)
  4717. {
  4718. return Ok(JsonView(false, groupData.Msg));
  4719. }
  4720. return Ok(JsonView(true, groupData.Msg, groupData.Data));
  4721. }
  4722. /// <summary>
  4723. /// 公务出访操作(Status:1.新增,2.修改)
  4724. /// </summary>
  4725. /// <param name="dto"></param>
  4726. /// <returns></returns>
  4727. [HttpPost]
  4728. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  4729. public async Task<IActionResult> OpOfficialActivities(OpOfficialActivitiesDto dto)
  4730. {
  4731. Result groupData = await _officialActivitiesRep.OpOfficialActivities(dto);
  4732. if (groupData.Code != 0)
  4733. {
  4734. return Ok(JsonView(StatusCodes.Status400BadRequest, groupData.Msg, new { Id = 0 }));
  4735. }
  4736. try
  4737. {
  4738. //公务出访变更发送通知
  4739. await AppNoticeLibrary.SendUserMsg_GroupShare_ToOP(dto.DiId, dto.CreateUserId);
  4740. }
  4741. catch (Exception ex)
  4742. {
  4743. //抄送日志
  4744. }
  4745. return Ok(JsonView(true, groupData.Msg, groupData.Data));
  4746. }
  4747. /// <summary>
  4748. /// 上传文件(邮件截图)
  4749. /// </summary>
  4750. /// <param name="dto"></param>
  4751. /// <returns></returns>
  4752. [HttpPost]
  4753. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  4754. public async Task<IActionResult> OfficialActivitiesUploadFiles([FromForm] OfficialActivitiesUploadFilesDto dto)
  4755. {
  4756. string networkPath = AppSettingsHelper.Get("GrpFileBaseUrl");
  4757. string localPath = AppSettingsHelper.Get("GrpFileBasePath");
  4758. string ptfPath = AppSettingsHelper.Get("GrpFileFtpPath");
  4759. if (dto.diId < 1 || dto.currUserId < 1)
  4760. {
  4761. return Ok(JsonView(false, "参数有误,上传失败!"));
  4762. }
  4763. if (dto.files == null || dto.files.Count < 1)
  4764. {
  4765. return Ok(JsonView(false, "文件为空,上传失败!"));
  4766. }
  4767. string localFileDir = $"{localPath}{ptfPath}";
  4768. List<string> fileUrls = new List<string>();
  4769. string[] failFiles = new string[] { };
  4770. foreach (var file in dto.files)
  4771. {
  4772. //文件名称
  4773. string[] fileNameArray = file.FileName.Split(".");
  4774. string projectFileName = $"{fileNameArray[0].ToString().Replace("-", "_").Replace("/", "_")}{DateTime.UtcNow:yyyyMMddHHmmss}.{fileNameArray[1]}";
  4775. //上传的文件的路径
  4776. string filePath = $@"{localPath}公务相关文件";
  4777. if (!Directory.Exists(filePath))
  4778. {
  4779. Directory.CreateDirectory(filePath);
  4780. }
  4781. var path = Path.Combine(filePath, projectFileName);
  4782. //using var stream = new FileStream(path, FileMode.Create);
  4783. //await file.CopyToAsync(stream);
  4784. try
  4785. {
  4786. using (FileStream fs = System.IO.File.Create(path))
  4787. {
  4788. file.CopyTo(fs);
  4789. fs.Flush();
  4790. }
  4791. fileUrls.Add($"/{ptfPath}公务相关文件/{projectFileName}");
  4792. }
  4793. catch (Exception ex)
  4794. {
  4795. failFiles.Append(file.FileName);
  4796. }
  4797. }
  4798. var oaInfo = await _sqlSugar.Queryable<Res_OfficialActivities>().Where(x => x.Id == dto.id).FirstAsync();
  4799. List<string> files = new List<string>();
  4800. var status = false;
  4801. var _oaInfo = new Res_OfficialActivities()
  4802. {
  4803. DiId = dto.diId,
  4804. ScreenshotOfMailUrl = JsonConvert.SerializeObject(fileUrls),
  4805. IsDel = 0,
  4806. CreateUserId = dto.currUserId,
  4807. CreateTime = DateTime.Now
  4808. };
  4809. if (oaInfo == null)
  4810. {
  4811. var id = await _sqlSugar.Insertable<Res_OfficialActivities>(_oaInfo)
  4812. .ExecuteReturnIdentityAsync();
  4813. if (id > 0)
  4814. {
  4815. status = true;
  4816. dto.id = id;
  4817. }
  4818. }
  4819. else
  4820. {
  4821. var oldFiles = string.IsNullOrEmpty(oaInfo.ScreenshotOfMailUrl)
  4822. ? new List<string>()
  4823. : JsonConvert.DeserializeObject<List<string>>(oaInfo.ScreenshotOfMailUrl);
  4824. if (oldFiles.Count > 0)
  4825. {
  4826. fileUrls.AddRange(oldFiles);
  4827. }
  4828. fileUrls = fileUrls.Distinct().ToList();
  4829. if (fileUrls.Count > 0)
  4830. {
  4831. var upd = await _sqlSugar.Updateable<Res_OfficialActivities>()
  4832. .SetColumns(x => x.ScreenshotOfMailUrl == JsonConvert.SerializeObject(fileUrls))
  4833. .Where(x => x.Id == dto.id)
  4834. .ExecuteCommandAsync();
  4835. if (upd > 0) status = true;
  4836. }
  4837. }
  4838. List<string> returnFileUrls = new List<string>();
  4839. fileUrls.ForEach(x =>
  4840. {
  4841. returnFileUrls.Add($"{networkPath}{x}");
  4842. });
  4843. if (status)
  4844. {
  4845. string msg = "操作成功!";
  4846. if (failFiles.Length > 1)
  4847. {
  4848. foreach (var item in failFiles)
  4849. {
  4850. msg += $"{failFiles}、";
  4851. }
  4852. msg = msg.Substring(0, msg.Length - 1);
  4853. msg += "等文件保存失败!";
  4854. }
  4855. return Ok(JsonView(true, msg, new { id = dto.id, fileUrls = returnFileUrls }));
  4856. }
  4857. return Ok(JsonView(false, "操作失败!"));
  4858. }
  4859. /// <summary>
  4860. /// 删除文件(邮件截图)
  4861. /// </summary>
  4862. /// <param name="dto"></param>
  4863. /// <returns></returns>
  4864. [HttpPost]
  4865. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  4866. public async Task<IActionResult> OfficialActivitiesDelFile(OfficialActivitiesDelFileDto dto)
  4867. {
  4868. string networkPath = AppSettingsHelper.Get("GrpFileBaseUrl");
  4869. string localPath = AppSettingsHelper.Get("GrpFileBasePath");
  4870. string ptfPath = AppSettingsHelper.Get("GrpFileFtpPath");
  4871. if (dto.Id < 1 || string.IsNullOrEmpty(dto.FileName))
  4872. {
  4873. return Ok(JsonView(false, "参数有误,上传失败!"));
  4874. }
  4875. string localFileDir = $"{localPath}{ptfPath}";
  4876. List<string> fileUrls = new List<string>();
  4877. var oaInfo = await _sqlSugar.Queryable<Res_OfficialActivities>().Where(x => x.Id == dto.Id).FirstAsync();
  4878. if (oaInfo != null)
  4879. {
  4880. string urlJson = oaInfo.ScreenshotOfMailUrl ?? "";
  4881. if (!string.IsNullOrEmpty(urlJson))
  4882. {
  4883. List<string> urls = JsonConvert.DeserializeObject<List<string>>(urlJson);
  4884. var filePath = urls.Find(x => x.Contains(dto.FileName));
  4885. if (string.IsNullOrEmpty(filePath)) return Ok(JsonView(false, "文件不存在!"));
  4886. var updFile = urls.Remove(filePath);
  4887. if (updFile)
  4888. {
  4889. //删除文件
  4890. string fileUrl = AppSettingsHelper.Get("GrpFileBasePath").Replace(@"/Office/GrpFile", "") + filePath;
  4891. if (System.IO.File.Exists(fileUrl))
  4892. {
  4893. System.IO.File.Delete(fileUrl);
  4894. }
  4895. string urlJsonstr = JsonConvert.SerializeObject(urls);
  4896. var upd = await _sqlSugar.Updateable<Res_OfficialActivities>()
  4897. .SetColumns(x => x.ScreenshotOfMailUrl == urlJsonstr)
  4898. .Where(x => x.Id == dto.Id)
  4899. .ExecuteCommandAsync();
  4900. if (upd > 0) return Ok(JsonView(true, "操作成功!"));
  4901. }
  4902. }
  4903. }
  4904. return Ok(JsonView(false, "操作失败!"));
  4905. }
  4906. /// <summary>
  4907. /// 公务出访 确认、取消邀请
  4908. /// </summary>
  4909. /// <param name="dto"></param>
  4910. /// <returns></returns>
  4911. [HttpPost]
  4912. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  4913. public async Task<IActionResult> OfficialActivitiesInviteOperation(OfficialActivitiesInviteOperationDto dto)
  4914. {
  4915. if (dto.Id < 1) return Ok(JsonView(false, "Id参数有误!"));
  4916. if (dto.Type < 0 || dto.Type > 1) return Ok(JsonView(false, "Type参数有误!"));
  4917. var upd = await _sqlSugar.Updateable<Res_OfficialActivities>()
  4918. .SetColumns(x => x.ConfirmTheInvitation == dto.Type)
  4919. .Where(x => x.Id == dto.Id)
  4920. .ExecuteCommandAsync();
  4921. if (upd > 0) return Ok(JsonView(true, "操作成功!"));
  4922. return Ok(JsonView(false, "操作失败!"));
  4923. }
  4924. /// <summary>
  4925. /// 上传文件
  4926. /// </summary>
  4927. /// <param name="file"></param>
  4928. /// <returns></returns>
  4929. [HttpPost]
  4930. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  4931. public async Task<IActionResult> UploadOfficialActivities(IFormFile file)
  4932. {
  4933. try
  4934. {
  4935. int Type = int.Parse(Request.Headers["Type"]);//1公务方简介,2公务活动图片,3发票
  4936. int DiId = int.Parse(Request.Headers["DiId"]);
  4937. int CreateUserId = int.Parse(Request.Headers["CreateUserId"]);
  4938. if (file != null)
  4939. {
  4940. var fileDir = AppSettingsHelper.Get("GrpFileBasePath");
  4941. //文件名称
  4942. string projectFileName = file.FileName;
  4943. //上传的文件的路径
  4944. string filePath = "";
  4945. if (!Directory.Exists(fileDir))
  4946. {
  4947. Directory.CreateDirectory(fileDir);
  4948. }
  4949. //上传的文件的路径
  4950. filePath = fileDir + $@"\商邀相关文件\{projectFileName}";
  4951. using (FileStream fs = System.IO.File.Create(filePath))
  4952. {
  4953. file.CopyTo(fs);
  4954. fs.Flush();
  4955. }
  4956. Grp_DeleFile d = new Grp_DeleFile();
  4957. d.Diid = DiId;
  4958. d.Category = 970;
  4959. if (Type == 1) d.Kind = 1;
  4960. else if (Type == 2) d.Kind = 2;
  4961. else if (Type == 3) d.Kind = 3;
  4962. d.FileName = projectFileName;
  4963. d.FilePath = "";
  4964. d.CreateUserId = CreateUserId;
  4965. int id = await _sqlSugar.Insertable(d).ExecuteReturnIdentityAsync();
  4966. return Ok(JsonView(true, "上传成功!", projectFileName));
  4967. }
  4968. else
  4969. {
  4970. return Ok(JsonView(false, "上传失败!"));
  4971. }
  4972. }
  4973. catch (Exception ex)
  4974. {
  4975. return Ok(JsonView(false, "程序错误!"));
  4976. throw;
  4977. }
  4978. }
  4979. /// <summary>
  4980. /// 删除文件
  4981. /// </summary>
  4982. /// <param name="dto"></param>
  4983. /// <returns></returns>
  4984. [HttpPost]
  4985. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  4986. public async Task<IActionResult> DelloadOfficialActivities(DelBaseDto dto)
  4987. {
  4988. try
  4989. {
  4990. var fileDir = AppSettingsHelper.Get("GrpFileBasePath");
  4991. Grp_DeleFile _DeleFile = await _sqlSugar.Queryable<Grp_DeleFile>().FirstAsync(a => a.Id == dto.Id);
  4992. if (_DeleFile != null)
  4993. {
  4994. string fileName = _DeleFile.FileName;
  4995. string filePath = fileDir + "/团组增减款项相关文件/" + fileName;
  4996. // 删除该文件
  4997. System.IO.File.Delete(filePath);
  4998. int id = await _sqlSugar.Updateable<Grp_DeleFile>().Where(a => a.Id == dto.Id).SetColumns(a => new Grp_DeleFile { IsDel = 1, DeleteUserId = dto.DeleteUserId, DeleteTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss") }).ExecuteCommandAsync();
  4999. return Ok(JsonView(true, "取消文件成功!"));
  5000. }
  5001. else
  5002. {
  5003. return Ok(JsonView(false, "取消文件失败!"));
  5004. }
  5005. }
  5006. catch (Exception ex)
  5007. {
  5008. return Ok(JsonView(false, "程序错误!"));
  5009. throw;
  5010. }
  5011. }
  5012. /// <summary>
  5013. /// 删除公务出访信息
  5014. /// </summary>
  5015. /// <param name="dto"></param>
  5016. /// <returns></returns>
  5017. [HttpPost]
  5018. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  5019. public async Task<IActionResult> DelOfficialActivities(DelBaseDto dto)
  5020. {
  5021. _sqlSugar.BeginTran();
  5022. var res = await _officialActivitiesRep.SoftDeleteByIdAsync<Res_OfficialActivities>(dto.Id.ToString(), dto.DeleteUserId);
  5023. if (!res)
  5024. {
  5025. _sqlSugar.RollbackTran();
  5026. return Ok(JsonView(false, "删除失败"));
  5027. }
  5028. #region 删除公务出访
  5029. await _sqlSugar.Updateable<Grp_OfficialDutyLinkTranslator>()
  5030. .SetColumns(x => new Grp_OfficialDutyLinkTranslator()
  5031. {
  5032. DeleteUserId = dto.DeleteUserId,
  5033. DeleteTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  5034. IsDel = 1,
  5035. Remark = $"公务出访-->删除"
  5036. })
  5037. .Where(x => x.OfficialDutyId == dto.Id)
  5038. .ExecuteCommandAsync();
  5039. #endregion
  5040. _sqlSugar.CommitTran();
  5041. return Ok(JsonView(true, "删除成功!"));
  5042. }
  5043. /// <summary>
  5044. /// 公务出访
  5045. /// 请示范例提示
  5046. /// Add Time:2024-05-13 13:56:44
  5047. /// </summary>
  5048. /// <param name="dto"></param>
  5049. /// <returns></returns>
  5050. [HttpPost]
  5051. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  5052. public async Task<IActionResult> PostOfficialActivitiesReqReqSampleTips(PostOfficialActivitiesReqReqSampleTipsDto dto)
  5053. {
  5054. var res = await _officialActivitiesRep.PostReqReqSampleTips(dto.country, dto.area, dto.client);
  5055. if (res.Code == 0)
  5056. {
  5057. return Ok(JsonView(true, "操作成功!", res.Data));
  5058. }
  5059. return Ok(JsonView(false, res.Msg));
  5060. }
  5061. /// <summary>
  5062. /// 公务出访 (省外办,市外办) File Downlaod
  5063. /// </summary>
  5064. /// <param name="dto"></param>
  5065. /// <returns></returns>
  5066. [HttpPost]
  5067. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  5068. public async Task<IActionResult> OfficialActivitiesFileDownload(OfficialActivitiesFileDownload dto)
  5069. {
  5070. #region 参数验证
  5071. if (dto.FileType < 1 || dto.FileType > 2) return Ok(JsonView(false, "请传入有效的FileType参数. 1 省外办出访请示 2 市外办出访请示", ""));
  5072. if (dto.DiId < 1) return Ok(JsonView(false, "请传入有效的DiId参数.", ""));
  5073. #endregion
  5074. //团组基础信息
  5075. var groupInfo = _sqlSugar.Queryable<Grp_DelegationInfo>().Where(it => it.IsDel == 0 && it.Id == dto.DiId).First();
  5076. if (groupInfo == null) return Ok(JsonView(false, "该团组基本信息不存在", ""));
  5077. groupInfo.VisitCountry = groupInfo.VisitCountry.Replace("|", "、");
  5078. //团组公务信息
  5079. var obDatas = _sqlSugar.Queryable<Res_OfficialActivities>().Where(it => it.IsDel == 0 && it.DiId == dto.DiId).OrderBy(it => it.Date).ToList();
  5080. //if (obDatas.Count < 1) return Ok(JsonView(false, "请先录入公务信息!"));
  5081. //团组客户名单
  5082. var guestResult = _tourClientListRep._ItemByDiId(1, dto.DiId);
  5083. List<TourClientListByDiIdView> guestInfos = new List<TourClientListByDiIdView>();
  5084. if (guestResult.Result.Code == 0) guestInfos = guestResult.Result.Data as List<TourClientListByDiIdView>;
  5085. string visitCountrys = ""; //××国家(或地区)×天,××国家(或地区)×天
  5086. string countryStayStr = ""; // xx、xx、xx
  5087. //出入境费用 住宿类型
  5088. var dayCostDatas = _sqlSugar.Queryable<Grp_DayAndCost>().Where(it => it.IsDel == 0 && it.DiId == dto.DiId && it.Type == 1).ToList();
  5089. if (dayCostDatas.Count > 0)
  5090. {
  5091. List<int> nationalIds = dayCostDatas.Select(it => it.NationalTravelFeeId).ToList();
  5092. var nationalDatas = _sqlSugar.Queryable<Grp_NationalTravelFee>().Where(it => nationalIds.Contains(it.Id)).ToList();
  5093. var nationalDataGroupByCountry = nationalDatas.GroupBy(it => it.Country);
  5094. foreach (var item in nationalDataGroupByCountry)
  5095. {
  5096. visitCountrys += $"{item.Key}、";
  5097. int stayDays = nationalIds.Where(it => it == item.ToList()[0].Id).Count();
  5098. countryStayStr += $"{item.Key}{stayDays}天,";
  5099. }
  5100. if (visitCountrys.Length > 0) visitCountrys = visitCountrys.Substring(0, visitCountrys.Length - 1);
  5101. if (countryStayStr.Length > 0) countryStayStr = countryStayStr.Substring(0, countryStayStr.Length - 1);
  5102. }
  5103. else
  5104. {
  5105. visitCountrys = groupInfo.VisitCountry;
  5106. countryStayStr = GeneralMethod.GetCountryStandingTime(groupInfo.Id); //计算国家出访天数
  5107. }
  5108. //出访人数
  5109. int visitPeopleNum = groupInfo.VisitPNumber;
  5110. int visitDaysNum = groupInfo.VisitDays;
  5111. //出访单位
  5112. string obInfoStr = "";
  5113. foreach (var ob in obDatas) obInfoStr += @$"{ob.Client.Trim()}{ob.Job.Trim()}{ob.Contact.Trim()}、";
  5114. if (obInfoStr.Length > 0) obInfoStr = obInfoStr.Substring(0, obInfoStr.Length - 1);
  5115. obInfoStr = obInfoStr ?? "[公务出访未录入]";
  5116. //出访路线
  5117. string lineStr = GeneralMethod.GetGroupCityLine(groupInfo.Id, "—");
  5118. if (dto.FileType == 1)
  5119. {
  5120. //载入模板
  5121. string tempPath = AppSettingsHelper.Get("WordBasePath") + "Template/省外办出访请示 - 模板.docx";
  5122. var doc = new Document(tempPath);
  5123. DocumentBuilder builder = new DocumentBuilder(doc);
  5124. //键值对存放数据
  5125. Dictionary<string, string> dic = new Dictionary<string, string>();
  5126. //××(组团单位):接团客户信息团组
  5127. dic.Add("GroupClient", $"{groupInfo.ClientUnit}\r\n");
  5128. //关于××(职务、姓名)等×人赴××(国家或地区)进行×××(出访目的)的请示
  5129. string guestName = "";
  5130. string guestJob = "";
  5131. string guestInfoStr = "";
  5132. var guestFirstInfo = guestInfos.FirstOrDefault();
  5133. if (guestFirstInfo != null)
  5134. {
  5135. guestName = guestFirstInfo.LastName.Trim() + guestFirstInfo.FirstName.Trim();
  5136. guestJob = guestFirstInfo.Job.Trim();
  5137. guestInfoStr = $@"{guestJob}、{guestName}";
  5138. }
  5139. guestInfoStr = guestInfoStr ?? "[接团客户名单未录入]";
  5140. string askTitle = $@"关于{guestInfoStr}等{visitPeopleNum}人赴{visitCountrys}进行{groupInfo.VisitPurpose}的请示";
  5141. dic.Add("AskTitle", askTitle);
  5142. //应×××(邀请方名称+邀请人职务和姓名)的邀请,我单位拟派×××(职务、姓名)等×人(人数)于×××年×××月×××日赴×××(国家或地区)进行×××(出访目的)。现请示如下。
  5143. string visitDateStr = @$"{groupInfo.VisitDate.Year}年{groupInfo.VisitDate.Month}月{groupInfo.VisitDate.Day}日";
  5144. string askSubTitle = $@"应{obInfoStr}的邀请,我单位拟派{guestInfoStr}等{visitPeopleNum}人于{visitDateStr}赴{visitCountrys}进行{groupInfo.VisitPurpose}。现请示如下。";
  5145. dic.Add("AskSubTitle", askSubTitle);
  5146. //出访目的
  5147. dic.Add("VisitPurpose", groupInfo.VisitPurpose);
  5148. //出访任务
  5149. //(一)××(国家或地区)
  5150. //1.拜访×××(机构名称)×××(姓名、职务)洽谈(或调研、推动等)×××(项目或机构名称等)。(例:拜会×××经济与发展司司长×××,商讨“×××活动”相关筹备工作。)
  5151. //2.拜访×××(机构名称)×××(姓名、职务)洽谈(或调研、推动等)×××(项目或机构名称等)。(例:拜会×××经济与发展司司长×××,商讨“×××活动”相关筹备工作。)
  5152. string obBackgroundStr = "";//出访背景
  5153. string taskStr = ""; ; //出访任务
  5154. var countrys = obDatas.GroupBy(it => it.Country);
  5155. int countryIndex = 1;
  5156. foreach (var item in countrys)
  5157. {
  5158. string taskTitle = $"({GetToUpperNumber(countryIndex)}){item.Key}\r\n";
  5159. string taskContent = "";
  5160. string obBackgroundContent = "";
  5161. int obIndex = 1;
  5162. foreach (var obInfo in item.ToList())
  5163. {
  5164. string reqSmaple = "";
  5165. string settingStr = "";
  5166. if (!string.IsNullOrEmpty(obInfo.ReqSample)) reqSmaple = obInfo.ReqSample;
  5167. else reqSmaple = "[公务出访请示范例未录入]";
  5168. if (!string.IsNullOrEmpty(obInfo.Setting)) settingStr = obInfo.Setting;
  5169. else settingStr = "[公务方背景未录入]";
  5170. taskContent += $"{obIndex}. {reqSmaple}\r\n";
  5171. obBackgroundContent += $"{obIndex}. {obInfo.Client}:{settingStr}\r\n";
  5172. obIndex++;
  5173. }
  5174. taskStr += $"{taskTitle}{taskContent}";
  5175. obBackgroundStr += $"{taskTitle}{obBackgroundContent}";
  5176. countryIndex++;
  5177. }
  5178. //出访背景
  5179. if (obBackgroundStr.Length < 1) obBackgroundStr = "[公务出访背景未录入]";
  5180. dic.Add("OBSetting", obBackgroundStr);
  5181. //出访任务
  5182. if (taskStr.Length < 1) taskStr = "[公务出访任务未录入]";
  5183. dic.Add("TaskContent", taskStr);
  5184. //出访时间
  5185. //代表团拟于××年×月×日—×月×日出访,在外停留×天。其中,××国家(或地区)×天,××国家(或地区)×天。
  5186. string visitTimeQuantumStr = $"代表团拟于{groupInfo.VisitStartDate.Year}年{groupInfo.VisitStartDate.Month}月{groupInfo.VisitStartDate.Day}日—{groupInfo.VisitEndDate.Month}月{groupInfo.VisitEndDate.Day}日出访,在外停留{visitDaysNum}天。其中,{countryStayStr}。";
  5187. //出访时间
  5188. dic.Add("TimeQuantum", visitTimeQuantumStr);
  5189. //出访路线
  5190. //成都—××(出境城市名称)—××(转机不出机场)—××(公务所在城市)……—××(入境城市名称)—成都。(例:成都—法兰克福<转机不出机场>—巴黎—巴塞罗那—阿姆斯特丹<转机不出机场>-成都)
  5191. dic.Add("Line", lineStr);
  5192. //scheduling //行程安排
  5193. string schedulingStr = $"[OP行程单暂未生成]";
  5194. #region op行程 根据黑屏代码录入
  5195. var opTripView = GeneralMethod.GetBriefStroke(groupInfo.Id);
  5196. if (opTripView.Code != 0) schedulingStr = $"[{opTripView.Msg}]";
  5197. schedulingStr = "";
  5198. List<Grp_TravelList> travelList = new List<Grp_TravelList>();
  5199. travelList = opTripView.Data as List<Grp_TravelList>;
  5200. if (travelList != null)
  5201. {
  5202. foreach (var item in travelList)
  5203. {
  5204. schedulingStr += $"{item.Date}({item.WeekDay})\r\n{item.Trip}\r\n";
  5205. }
  5206. }
  5207. #endregion
  5208. dic.Add("Scheduling", schedulingStr);
  5209. #region 填充word模板书签内容
  5210. foreach (var key in dic.Keys)
  5211. {
  5212. builder.MoveToBookmark(key);
  5213. builder.Write(dic[key]);
  5214. }
  5215. #endregion
  5216. //获取word里所有表格
  5217. NodeCollection allTables = doc.GetChildNodes(NodeType.Table, true);
  5218. //获取所填表格的序数
  5219. Aspose.Words.Tables.Table tableOne = allTables[0] as Aspose.Words.Tables.Table;
  5220. var rowStart = tableOne.Rows[0]; //获取第1行
  5221. //循环赋值
  5222. for (int i = 0; i < guestInfos.Count; i++)
  5223. {
  5224. var guestInfo = guestInfos[i];
  5225. builder.MoveToCell(0, i + 1, 0, 0);
  5226. builder.Write(guestInfo.LastName + guestInfo.FirstName);
  5227. builder.MoveToCell(0, i + 1, 1, 0);
  5228. int sex = guestInfo.Sex;
  5229. string sexStr = string.Empty;
  5230. if (sex == 0) sexStr = "男";
  5231. else if (sex == 1) sexStr = "女";
  5232. else sexStr = "-";
  5233. builder.Write(sexStr);
  5234. builder.MoveToCell(0, i + 1, 2, 0);
  5235. builder.Write(guestInfo.CompanyFullName + guestInfo.Job);
  5236. builder.MoveToCell(0, i + 1, 3, 0);
  5237. //string birthDay = "";
  5238. string birthDayStr = string.Empty;
  5239. if (guestInfo.BirthDay != DateTime.MinValue)
  5240. {
  5241. birthDayStr = guestInfo.BirthDay.ToString("yyyy.MM");
  5242. }
  5243. builder.Write(birthDayStr);
  5244. }
  5245. //删除多余行
  5246. while (tableOne.Rows.Count > guestInfos.Count + 1)
  5247. {
  5248. tableOne.Rows.RemoveAt(guestInfos.Count + 1);
  5249. }
  5250. var fileDir = AppSettingsHelper.Get("WordBasePath");
  5251. string fileName = $"{groupInfo.TeamName}省外办出访请示{DateTime.Now.ToString("yyyyMMddHHmmss")}.docx";
  5252. string filePath = fileDir + $@"OfficialActivities/{fileName}";
  5253. doc.Save(filePath);
  5254. string Url = $@"{AppSettingsHelper.Get("WordBaseUrl")}Office/Word/OfficialActivities/{fileName}";
  5255. return Ok(JsonView(true, "操作成功!", Url));
  5256. }
  5257. else if (dto.FileType == 2)
  5258. {
  5259. //载入模板
  5260. string tempPath = AppSettingsHelper.Get("WordBasePath") + "Template/市外办出访请示 - 模板.docx";
  5261. var doc = new Document(tempPath);
  5262. DocumentBuilder builder = new DocumentBuilder(doc);
  5263. //键值对存放数据
  5264. var dic = new Dictionary<string, string>
  5265. {
  5266. //××(组团单位):接团客户信息团组
  5267. { "GroupClient", $"{groupInfo.ClientUnit}" },
  5268. { "GroupClient1", $"{groupInfo.ClientUnit}" }
  5269. };
  5270. //关于××(职务、姓名)等×人赴××(国家或地区)进行×××(出访目的)的请示
  5271. string guestName = "";
  5272. string guestJob = "";
  5273. string guestInfoStr = "";
  5274. var guestFirstInfo = guestInfos.FirstOrDefault();
  5275. if (guestFirstInfo != null)
  5276. {
  5277. guestName = guestFirstInfo.LastName.Trim() + guestFirstInfo.FirstName.Trim();
  5278. guestJob = guestFirstInfo.Job;
  5279. guestInfoStr = $@"{guestJob}、{guestName.Trim()}";
  5280. }
  5281. string reqTitle = $@"关于{guestInfoStr}等{visitPeopleNum}人赴{visitCountrys}进行{groupInfo.VisitPurpose}的请示";
  5282. dic.Add("ReqTitle", reqTitle);
  5283. //应×××(邀请方名称+邀请人职务和姓名)的邀请,我单位拟派×××(职务、姓名)等×人(人数)于×××年×××月×××日赴×××(国家或地区)进行×××(出访目的)。
  5284. string visitDateStr = @$"{groupInfo.VisitDate.Year}年{groupInfo.VisitDate.Month}月{groupInfo.VisitDate.Day}日";
  5285. string reqSubTitle = $@"应{obInfoStr}的邀请,我单位拟派{guestInfoStr}等{visitPeopleNum}人于{visitDateStr}赴{visitCountrys}进行{groupInfo.VisitPurpose}。";
  5286. dic.Add("ReqSubTitle", reqSubTitle);
  5287. /*
  5288. * 出访任务
  5289. * (一)××(国家或地区)
  5290. * 1.拜访×××(机构名称)×××(姓名、职务)洽谈(或调研、推动等)×××(项目或机构名称等)。(例:拜会×××经济与发展司司长×××,商讨“×××活动”相关筹备工作。)背景:拟拜访机构的优势、拟洽谈项目的前期进展情况等。(例:背景:由×××总领馆联合×××市政府共同举办的“×××活动”将于今年6月在×××举行。)
  5291. * 2.拜访×××(机构名称)×××(姓名、职务)洽谈(或调研等)×××(项目或机构名称等)。(例:拜会友城×××市市长×××,巩固和发展两市传统友谊,商谈两市未来互动交流合作项目。)背景:拟拜访机构的优势、拟洽谈项目的前期进展情况等。(例:背景:×××年×月,×××市与×××市正式缔结友好城市关系。在此框架下,两市开展了一系列友好互访和商务交流。并于×××年×月共同举办了“×××活动”。)
  5292. *
  5293. */
  5294. string taskStr = string.Empty;
  5295. List<string> countrys = obDatas.Where(it => !string.IsNullOrEmpty(it.Country)).Select(it => it.Country).ToList();
  5296. int countryIndex = 1;
  5297. foreach (var item in countrys)
  5298. {
  5299. string taskTitle = $"({GetToUpperNumber(countryIndex)}){item ?? "[公务出访国家未填写]"}\r\n";
  5300. string taskContent = "";
  5301. if (!string.IsNullOrEmpty(item))
  5302. {
  5303. var countryObDatas = obDatas.Where(it => !string.IsNullOrEmpty(it.Country) && it.Country.Contains(item)).OrderBy(it => it.Date).ToList();
  5304. int obIndex = 1;
  5305. if (countryObDatas.Count == 0) taskContent = "[公务出访未录入]\r\n";
  5306. else
  5307. {
  5308. foreach (var obInfo in countryObDatas)
  5309. {
  5310. taskContent += $"{obIndex}.{obInfo.ReqSample ?? "[公务出访请示范例未填写]"}\r\n背景:{obInfo.Setting ?? "[公务出访背景未填写]"}\r\n";
  5311. obIndex++;
  5312. }
  5313. }
  5314. }
  5315. taskStr += $"{taskTitle}{taskContent}";
  5316. countryIndex++;
  5317. }
  5318. if (string.IsNullOrEmpty(taskStr)) taskStr = "××××××";
  5319. //出访任务
  5320. dic.Add("TaskContent", taskStr);
  5321. //出访目的
  5322. dic.Add("VisitPurpose", groupInfo.VisitPurpose);
  5323. //出访信息
  5324. //代表团拟于××年×月×日—×月×日出访,在外停留×天。其中,××国家(或地区)×天,××国家(或地区)×天。出访路线:成都—××(出境城市名称)—××(转机不出机场)—××(公务所在城市)……—××(入境城市名称)—成都。(例:成都—法兰克福<转机不出机场>—巴黎—巴塞罗那—阿姆斯特丹<转机不出机场>-成都)出访费用:由××承担(注明由外方或上级机关承担,还是由派员单位在年度安排的预算经费中列支)。
  5325. string tripInfoStr = "";
  5326. tripInfoStr = $"代表团拟于{groupInfo.VisitStartDate.Year}年{groupInfo.VisitStartDate.Month}月{groupInfo.VisitStartDate.Day}日—{groupInfo.VisitEndDate.Month}月{groupInfo.VisitEndDate.Day}日出访,在外停留{visitDaysNum}天。其中,{countryStayStr}。出访路线:{lineStr}。出访费用:由××承担(注明由外方或上级机关承担,还是由派员单位在年度安排的预算经费中列支)";
  5327. dic.Add("TripInfo", tripInfoStr);
  5328. #region 填充word模板书签内容
  5329. foreach (var key in dic.Keys)
  5330. {
  5331. builder.MoveToBookmark(key);
  5332. builder.Write(dic[key]);
  5333. }
  5334. #endregion
  5335. //获取word里所有表格
  5336. NodeCollection allTables = doc.GetChildNodes(NodeType.Table, true);
  5337. //获取所填表格的序数
  5338. Aspose.Words.Tables.Table tableOne = allTables[0] as Aspose.Words.Tables.Table;
  5339. var rowStart = tableOne.Rows[0]; //获取第1行
  5340. //循环赋值
  5341. for (int i = 0; i < guestInfos.Count; i++)
  5342. {
  5343. var guestInfo = guestInfos[i];
  5344. builder.MoveToCell(0, i + 1, 0, 0);
  5345. builder.Write(guestInfo.LastName + guestInfo.FirstName);
  5346. builder.MoveToCell(0, i + 1, 1, 0);
  5347. int sex = guestInfo.Sex;
  5348. string sexStr = string.Empty;
  5349. if (sex == 0) sexStr = "男";
  5350. else if (sex == 1) sexStr = "女";
  5351. else sexStr = "未设置";
  5352. builder.Write(sexStr);
  5353. builder.MoveToCell(0, i + 1, 2, 0);
  5354. builder.Write(guestInfo.CompanyFullName + guestInfo.Job);
  5355. builder.MoveToCell(0, i + 1, 3, 0);
  5356. DateTime birthDay = guestInfo.BirthDay;
  5357. string birthDayStr = string.Empty;
  5358. if (birthDay != DateTime.MinValue)
  5359. {
  5360. birthDayStr = birthDay.ToString("yyyy.MM");
  5361. }
  5362. builder.Write(birthDayStr);
  5363. }
  5364. //删除多余行
  5365. while (tableOne.Rows.Count > guestInfos.Count + 1)
  5366. {
  5367. tableOne.Rows.RemoveAt(guestInfos.Count + 1);
  5368. }
  5369. var fileDir = AppSettingsHelper.Get("WordBasePath");
  5370. var fileName = $"{groupInfo.TeamName}市外办出访请示{DateTime.Now:yyyyMMddHHmmss}.docx";
  5371. var filePath = fileDir + $@"OfficialActivities/{fileName}";
  5372. doc.Save(filePath);
  5373. var url = $@"{AppSettingsHelper.Get("WordBaseUrl")}Office/Word/OfficialActivities/{fileName}";
  5374. return Ok(JsonView(true, "操作成功!", url));
  5375. }
  5376. return Ok(JsonView(true, "操作失败!", ""));
  5377. }
  5378. private static string GetToUpperNumber(int num)
  5379. {
  5380. string numStr = "";
  5381. if (num == 1) numStr = "一";
  5382. else if (num == 2) numStr = "二";
  5383. else if (num == 3) numStr = "三";
  5384. else if (num == 4) numStr = "四";
  5385. else if (num == 5) numStr = "五";
  5386. else if (num == 6) numStr = "六";
  5387. else if (num == 7) numStr = "七";
  5388. else if (num == 8) numStr = "八";
  5389. else if (num == 9) numStr = "九";
  5390. else if (num == 10) numStr = "十";
  5391. return numStr;
  5392. }
  5393. [HttpPost]
  5394. public IActionResult ExportOfficialActivitiesInfo(ExportOfficialActivitiesInfoDto dto)
  5395. {
  5396. var jw = JsonView(false);
  5397. if (!ModelState.IsValid)
  5398. {
  5399. jw.Data = ModelState;
  5400. return Ok(jw);
  5401. }
  5402. var startTime_bool = DateTime.TryParse(dto.StartTime, out DateTime startTime_parse);
  5403. var endTime_bool = DateTime.TryParse(dto.EndTime, out DateTime endTime_parse);
  5404. if (startTime_parse > endTime_parse)
  5405. {
  5406. jw.Msg = "开始实际不能大于结束时间!";
  5407. return Ok(jw);
  5408. }
  5409. if (startTime_bool && endTime_bool)
  5410. {
  5411. //处理时间模板
  5412. var Query_DB = _sqlSugar.Queryable<Grp_DelegationInfo>()
  5413. .LeftJoin<Res_OfficialActivities>((a, b) => a.Id == b.DiId && b.IsDel == 0)
  5414. .LeftJoin<Sys_Users>((a, b, c) => b.CreateUserId == c.Id && c.IsDel == 0)
  5415. .Where((a, b, c) => a.IsDel == 0 && a.VisitDate >= startTime_parse && a.VisitDate <= endTime_parse)
  5416. .WhereIF(dto.State != -1, (a, b, c) => b.ConfirmTheInvitation == dto.State)
  5417. .Select((a, b, c) => new
  5418. {
  5419. a.TeamName,
  5420. b.Client,
  5421. b.ConfirmTheInvitation,
  5422. b.Contact,
  5423. ConfirmTheInvitationStr = b.ConfirmTheInvitation == 1 ? "是" : "否",
  5424. b.Date,
  5425. b.Nature,
  5426. c.CnName,
  5427. b.Address,
  5428. b.CreateTime
  5429. })
  5430. .ToList()
  5431. .Where(x => !string.IsNullOrWhiteSpace(x.Client))
  5432. .ToList();
  5433. if (Query_DB.Count > 0)
  5434. {
  5435. //打开excel
  5436. var ftpPath = AppSettingsHelper.Get("ExcelFtpPath");
  5437. var fileBasePath = AppSettingsHelper.Get("ExcelBasePath");
  5438. //var fileName = "公务出访信息.docx";
  5439. //创建数据源Table
  5440. DataTable dtSource = new DataTable();
  5441. dtSource.TableName = "TB";
  5442. var firstElem = Query_DB.First();
  5443. //遍历firstElem所有属性
  5444. foreach (var item in firstElem.GetType().GetProperties())
  5445. {
  5446. dtSource.Columns.Add(item.Name, item.PropertyType);
  5447. }
  5448. foreach (var item in Query_DB)
  5449. {
  5450. DataRow dr = dtSource.NewRow();
  5451. //遍历dtSource所有列头
  5452. foreach (DataColumn column in dtSource.Columns)
  5453. {
  5454. dr[column.ColumnName] = item.GetType().GetProperty(column.ColumnName)?.GetValue(item)?.ToString();
  5455. }
  5456. dtSource.Rows.Add(dr);
  5457. }
  5458. WorkbookDesigner designer = new WorkbookDesigner();
  5459. designer.Workbook = new Workbook(fileBasePath + ("Template/公务出访信息.xlsx"));
  5460. var title = startTime_parse.ToString("yyyy年MM月dd日") + " - " + endTime_parse.ToString("yyyy年MM月dd日") + " 公务出访信息";
  5461. designer.SetDataSource("title", title);
  5462. designer.SetDataSource(dtSource);
  5463. designer.Process();
  5464. var exportSerevePath = fileBasePath + "ExportOfficialActivitiesInfo/" + title + ".xlsx";
  5465. var exportFtpPaht = ftpPath + "ExportOfficialActivitiesInfo/" + title + ".xlsx";
  5466. //保存文件
  5467. designer.Workbook.Save(exportSerevePath);
  5468. jw.Data = AppSettingsHelper.Get("ExcelBaseUrl") + exportFtpPaht;
  5469. jw.Code = 200;
  5470. jw.Msg = "获取成功!";
  5471. }
  5472. else
  5473. {
  5474. jw.Data = "无公务信息!";
  5475. }
  5476. }
  5477. else
  5478. {
  5479. jw.Data = "时间格式错误!";
  5480. }
  5481. return Ok(jw);
  5482. }
  5483. #endregion
  5484. #region 请示数据库
  5485. /// <summary>
  5486. /// 查询请示数据库初始化
  5487. /// </summary>
  5488. /// <param name="dto"></param>
  5489. /// <returns></returns>
  5490. [HttpPost]
  5491. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  5492. public async Task<IActionResult> QueryAskData(QueryAskDataDto dto)
  5493. {
  5494. Result groupData = await _askDataRep.QueryAskData(dto);
  5495. if (groupData.Code != 0)
  5496. {
  5497. return Ok(JsonView(false, groupData.Msg));
  5498. }
  5499. return Ok(JsonView(true, groupData.Msg, groupData.Data));
  5500. }
  5501. /// <summary>
  5502. /// 根据Id查询请示数据库单条数据
  5503. /// </summary>
  5504. /// <param name="dto"></param>
  5505. /// <returns></returns>
  5506. [HttpPost]
  5507. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  5508. public async Task<IActionResult> QueryAskDataById(QueryAskDataByIdDto dto)
  5509. {
  5510. Res_AskData _AskData = await _sqlSugar.Queryable<Res_AskData>().FirstAsync(a => a.IsDel == 0 && a.Id == dto.id);
  5511. if (_AskData == null)
  5512. {
  5513. return Ok(JsonView(true, "暂无数据!", _AskData));
  5514. }
  5515. return Ok(JsonView(true, "查询成功!", _AskData));
  5516. }
  5517. /// <summary>
  5518. /// 请示数据库操作(Status:1.新增,2.修改)
  5519. /// </summary>
  5520. /// <param name="dto"></param>
  5521. /// <returns></returns>
  5522. [HttpPost]
  5523. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  5524. public async Task<IActionResult> OpAskData(OpAskDataDto dto)
  5525. {
  5526. Result groupData = await _askDataRep.OpAskData(dto);
  5527. if (groupData.Code != 0)
  5528. {
  5529. return Ok(JsonView(false, groupData.Msg));
  5530. }
  5531. return Ok(JsonView(true, groupData.Msg, groupData.Data));
  5532. }
  5533. /// <summary>
  5534. /// 删除请示资料信息
  5535. /// </summary>
  5536. /// <param name="dto"></param>
  5537. /// <returns></returns>
  5538. [HttpPost]
  5539. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  5540. public async Task<IActionResult> DelAskData(DelBaseDto dto)
  5541. {
  5542. var res = await _askDataRep.SoftDeleteByIdAsync<Res_AskData>(dto.Id.ToString(), dto.DeleteUserId);
  5543. if (!res)
  5544. {
  5545. return Ok(JsonView(false, "删除失败"));
  5546. }
  5547. return Ok(JsonView(true, "删除成功!"));
  5548. }
  5549. /// <summary>
  5550. /// 请示资料信息
  5551. /// 省外办下载
  5552. /// </summary>
  5553. /// <param name="dto"></param>
  5554. /// <returns></returns>
  5555. [HttpPost]
  5556. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  5557. public async Task<IActionResult> AskDataProvinceFAOPFile(AskDataProvinceFileDto dto)
  5558. {
  5559. if (dto.DataIdItem.Length < 1) return Ok(JsonView(false, $"请传入有效的数据Id集合!"));
  5560. if (dto.DiId < 1) return Ok(JsonView(false, $"请传入有效的数据DiId参数!"));
  5561. //大写数字序号
  5562. string[] num = { "一", "二", "三", "四", "五", "六", "七", "八", "九", "十" ,
  5563. "十一","十二","十三","十四","十五","十六","十七","十八","十九","二十",
  5564. "二十一","二十二","二十三","二十四","二十五","二十六","二十七","二十八","二十九","三十",
  5565. "三十一","三十二","三十三","三十四","三十五","三十六","三十七","三十八","三十九","四十"};
  5566. //数据源
  5567. var idItem = dto.DataIdItem.OrderBy(x => x);
  5568. var askDatas = await _sqlSugar.Queryable<Res_AskData>().Where(x => x.IsDel == 0 && idItem.Contains(x.Id)).ToListAsync();
  5569. //书签名
  5570. string Area = "",
  5571. Official = "";
  5572. //表格数据源
  5573. DataTable dt = new DataTable();
  5574. dt.Columns.Add("Country", typeof(string));
  5575. dt.Columns.Add("TalkCase", typeof(string));
  5576. int id = 0;
  5577. int TalkcaseOrder = 0;
  5578. int RowOrder = 0;
  5579. //填充数据
  5580. for (int i = 0; i < askDatas.Count; i++)
  5581. {
  5582. var ad = askDatas[i];
  5583. Area = Area + ad.Country + ad.Area + "、";
  5584. Official = Official + ad.UnitName + "、";
  5585. //填充表格数据源
  5586. DataRow dr = dt.NewRow();
  5587. if (i > 0)
  5588. {
  5589. var adtemp = askDatas[i - 1];
  5590. if (ad.Country == adtemp.Country)
  5591. {
  5592. TalkcaseOrder++;
  5593. dt.Rows[RowOrder]["TalkCase"] = dt.Rows[RowOrder]["TalkCase"] + "\r\n" + (1 + TalkcaseOrder).ToString() + "、" + ad.TalkCase;
  5594. }
  5595. else
  5596. {
  5597. RowOrder++;
  5598. dr = dt.NewRow();
  5599. dr["Country"] = "(" + num[RowOrder] + ")" + ad.Country;
  5600. dr["TalkCase"] = "1、" + ad.TalkCase;
  5601. dt.Rows.Add(dr);
  5602. TalkcaseOrder = 0;
  5603. }
  5604. }
  5605. else
  5606. {
  5607. dr["Country"] = "(" + num[i] + ")" + ad.Country;
  5608. dr["TalkCase"] = "1、" + ad.TalkCase;
  5609. dt.Rows.Add(dr);
  5610. }
  5611. id = ad.Id;
  5612. }
  5613. //模板路径
  5614. var tempPath = AppSettingsHelper.Get("WordBasePath") + "Template/省外办出访请示模板.doc";
  5615. //载入模板
  5616. Aspose.Words.Document doc = new Aspose.Words.Document(tempPath);
  5617. #region 填充word内容
  5618. #region 填充请示内容
  5619. //利用DocumentBuilder处理表格或是书签
  5620. Aspose.Words.DocumentBuilder builder = new DocumentBuilder(doc);
  5621. string bookmark = "";
  5622. try
  5623. {
  5624. for (int i = 0; i < dt.Rows.Count; i++)
  5625. {
  5626. bookmark = bookmark
  5627. + dt.Rows[i]["Country"].ToString() + "\r\n"
  5628. + dt.Rows[i]["TalkCase"].ToString() + "\r\n";
  5629. }
  5630. }
  5631. catch
  5632. {
  5633. }
  5634. #endregion
  5635. #region 填充word模板书签内容
  5636. //团组信息
  5637. var di = await _sqlSugar.Queryable<Grp_DelegationInfo>().Where(x => x.IsDel == 0 && x.Id == dto.DiId).FirstAsync();
  5638. //接团客户名单实例
  5639. string client_sql = string.Format(@"
  5640. SELECT
  5641. tcl.Id,
  5642. tcl.DiId,
  5643. tcl.IsAccompany,
  5644. temp.*,
  5645. u.CnName AS Operator,
  5646. tcl.CreateTime AS OperatingTime
  5647. FROM
  5648. Grp_TourClientList tcl
  5649. LEFT JOIN (
  5650. SELECT
  5651. dc.Id AS DcId,
  5652. dc.LastName,
  5653. dc.FirstName,
  5654. ccom.CompanyFullName,
  5655. dc.Job,
  5656. cc.CertNo AS IDCardNo,
  5657. dc.Sex,
  5658. dc.BirthDay
  5659. FROM
  5660. Crm_DeleClient dc
  5661. LEFT JOIN Crm_CustomerCompany ccom ON dc.CrmCompanyId = ccom.Id
  5662. AND ccom.IsDel = 0
  5663. LEFT JOIN Crm_CustomerCert cc ON dc.Id = cc.DcId
  5664. AND cc.SdId = 773
  5665. AND cc.IsDel = 0
  5666. WHERE
  5667. dc.IsDel = 0
  5668. ) temp ON temp.DcId = tcl.ClientId
  5669. LEFT JOIN Sys_Users u ON tcl.CreateUserId = u.Id
  5670. WHERE
  5671. tcl.IsDel = 0
  5672. AND tcl.DiId = {0}", dto.DiId);
  5673. var clientList = await _sqlSugar.SqlQueryable<TourClientListDetailsView>(client_sql).ToListAsync();
  5674. //机票代码实例
  5675. DataTable airTable = GeneralMethod.GetTableByBlackCode(dto.DiId);
  5676. //三字码
  5677. var threeCodes = await _sqlSugar.Queryable<Res_ThreeCode>().Where(x => x.IsDel == 0).ToListAsync();
  5678. //团名
  5679. if (doc.Range.Bookmarks["TeamName"] != null)
  5680. {
  5681. Bookmark mark = doc.Range.Bookmarks["TeamName"];
  5682. mark.Text = di.TeamName;
  5683. }
  5684. //团长名
  5685. if (doc.Range.Bookmarks["ClientName2"] != null)
  5686. {
  5687. Bookmark mark = doc.Range.Bookmarks["ClientName2"];
  5688. if (clientList == null || clientList.Count == 0)
  5689. {
  5690. mark.Text = "【本团尚未录入客户名单,请联系主管!】";
  5691. }
  5692. else
  5693. {
  5694. mark.Text = clientList[0].CompanyFullName + clientList[0].Job + clientList[0].LastName + clientList[0].FirstName;
  5695. }
  5696. }
  5697. if (doc.Range.Bookmarks["ClientName"] != null)
  5698. {
  5699. Bookmark mark = doc.Range.Bookmarks["ClientName"];
  5700. if (clientList == null || clientList.Count == 0)
  5701. {
  5702. mark.Text = "【本团尚未录入客户名单,请联系主管!】";
  5703. }
  5704. else
  5705. {
  5706. mark.Text = clientList[0].CompanyFullName + clientList[0].Job + clientList[0].LastName + clientList[0].FirstName;
  5707. }
  5708. }
  5709. //跟团人数
  5710. if (doc.Range.Bookmarks["ClientNumber2"] != null)
  5711. {
  5712. Bookmark mark = doc.Range.Bookmarks["ClientNumber2"];
  5713. mark.Text = di.VisitPNumber.ToString();
  5714. }
  5715. if (doc.Range.Bookmarks["ClientNumber"] != null)
  5716. {
  5717. Bookmark mark = doc.Range.Bookmarks["ClientNumber"];
  5718. mark.Text = di.VisitPNumber.ToString();
  5719. }
  5720. //地区
  5721. if (doc.Range.Bookmarks["Area"] != null)
  5722. {
  5723. Bookmark mark = doc.Range.Bookmarks["Area"];
  5724. mark.Text = Area.TrimEnd('、');
  5725. }
  5726. if (doc.Range.Bookmarks["Area2"] != null)
  5727. {
  5728. Bookmark mark = doc.Range.Bookmarks["Area2"];
  5729. mark.Text = Area.TrimEnd('、');
  5730. }
  5731. //出访目的
  5732. if (doc.Range.Bookmarks["Destination2"] != null)
  5733. {
  5734. Bookmark mark = doc.Range.Bookmarks["Destination2"];
  5735. mark.Text = di.VisitPurpose;
  5736. }
  5737. if (doc.Range.Bookmarks["Destination"] != null)
  5738. {
  5739. Bookmark mark = doc.Range.Bookmarks["Destination"];
  5740. mark.Text = di.VisitPurpose;
  5741. }
  5742. //邀请方
  5743. if (doc.Range.Bookmarks["Official"] != null)
  5744. {
  5745. Bookmark mark = doc.Range.Bookmarks["Official"];
  5746. mark.Text = Official.TrimEnd('、');
  5747. }
  5748. //出访日期
  5749. if (doc.Range.Bookmarks["VisitDate"] != null)
  5750. {
  5751. Bookmark mark = doc.Range.Bookmarks["VisitDate"];
  5752. mark.Text = di.VisitStartDate.Year + "年"
  5753. + di.VisitStartDate.Month + "月"
  5754. + di.VisitStartDate.Day + "日"
  5755. + "至"
  5756. + di.VisitEndDate.Month + "月"
  5757. + di.VisitEndDate.Day + "日";
  5758. }
  5759. //请示范例
  5760. if (doc.Range.Bookmarks["Temp"] != null)
  5761. {
  5762. Bookmark mark = doc.Range.Bookmarks["Temp"];
  5763. mark.Text = bookmark;
  5764. }
  5765. //出访起止日期
  5766. if (doc.Range.Bookmarks["Duration"] != null)
  5767. {
  5768. Bookmark mark = doc.Range.Bookmarks["Duration"];
  5769. mark.Text = di.VisitStartDate.Year + "年"
  5770. + di.VisitStartDate.Month + "月"
  5771. + di.VisitStartDate.Day + "日"
  5772. + "至"
  5773. + di.VisitEndDate.Month + "月"
  5774. + di.VisitEndDate.Day + "日";
  5775. }
  5776. //出访天数
  5777. if (doc.Range.Bookmarks["Durationdays"] != null)
  5778. {
  5779. Bookmark mark = doc.Range.Bookmarks["Durationdays"];
  5780. mark.Text = di.VisitDays.ToString();
  5781. }
  5782. if (doc.Range.Bookmarks["Durationdays2"] != null)
  5783. {
  5784. Bookmark mark = doc.Range.Bookmarks["Durationdays2"];
  5785. mark.Text = di.VisitDays.ToString();
  5786. }
  5787. #region 根据黑屏代码获取国家呆的天数;获取路线
  5788. string result_CountryAndDay = "", result_Air = "";
  5789. if (airTable.Rows[airTable.Rows.Count - 1]["Day"].ToString() == "")
  5790. {
  5791. result_CountryAndDay = airTable.Rows[airTable.Rows.Count - 1]["Fliagtcode"].ToString();
  5792. result_Air = result_CountryAndDay;
  5793. }
  5794. else
  5795. {
  5796. string place_start, place_end;
  5797. DateTime time_start, time_end;
  5798. Res_ThreeCode three = null;
  5799. for (int i = 0; i < airTable.Rows.Count; i++)
  5800. {
  5801. //三字码实例
  5802. three = threeCodes.Find(x => x.Three == airTable.Rows[i]["Three"].ToString().Substring(0, 3));
  5803. if (three == null)
  5804. {
  5805. place_start = "【未收录三字码" + airTable.Rows[i]["Three"].ToString().Substring(0, 3) + ",请联系机票同事】";
  5806. }
  5807. else
  5808. {
  5809. if (string.IsNullOrEmpty(three.Country))
  5810. {
  5811. place_start = "【本三字码" + airTable.Rows[i]["Three"].ToString().Substring(0, 3) + "未录入国家信息,请联系机票同事】";
  5812. }
  5813. else
  5814. {
  5815. place_start = three.City;
  5816. }
  5817. }
  5818. three = threeCodes.Find(x => x.Three == airTable.Rows[i]["Three"].ToString().Substring(3, 3));
  5819. if (three == null)
  5820. {
  5821. place_end = "【未收录" + airTable.Rows[i]["Three"].ToString().Substring(0, 3) + ",请联系机票同事】";
  5822. }
  5823. else
  5824. {
  5825. if (string.IsNullOrEmpty(three.Country))
  5826. {
  5827. place_end = "【本三字码" + airTable.Rows[i]["Three"].ToString().Substring(0, 3) + "未录入国家信息,请联系机票同事】";
  5828. }
  5829. else
  5830. {
  5831. place_end = three.City;
  5832. }
  5833. }
  5834. result_Air += place_start + "一" + place_end + "一";
  5835. //获取国家和呆的天数
  5836. if (i > 0 && i < airTable.Rows.Count - 1)
  5837. {
  5838. time_start = Convert.ToDateTime(airTable.Rows[i - 1]["Day"]);
  5839. time_end = Convert.ToDateTime(airTable.Rows[i]["Day"]);
  5840. TimeSpan daycount = time_end.Subtract(time_start);
  5841. result_CountryAndDay += place_start + daycount.Days + "天" + ",";
  5842. }
  5843. }
  5844. }
  5845. #endregion
  5846. //每个国家呆的天数
  5847. if (doc.Range.Bookmarks["CountryAndDay"] != null)
  5848. {
  5849. Bookmark mark = doc.Range.Bookmarks["CountryAndDay"];
  5850. mark.Text = result_CountryAndDay.TrimEnd(',');
  5851. }
  5852. #region 出访路线城市字符串去重
  5853. result_Air = result_Air.TrimEnd('一');
  5854. string[] airTemp = result_Air.Split('一');
  5855. airTemp = airTemp.Distinct().ToArray();
  5856. result_Air = "";
  5857. foreach (string str in airTemp)
  5858. {
  5859. result_Air += str + "一";
  5860. }
  5861. result_Air = result_Air + airTemp[0];
  5862. #endregion
  5863. //出访路线
  5864. if (doc.Range.Bookmarks["Air"] != null)
  5865. {
  5866. Bookmark mark = doc.Range.Bookmarks["Air"];
  5867. mark.Text = result_Air;
  5868. }
  5869. #endregion
  5870. #region 客户名单
  5871. NodeCollection allTables = doc.GetChildNodes(NodeType.Table, true);
  5872. Aspose.Words.Tables.Table tableClient = allTables[0] as Aspose.Words.Tables.Table;
  5873. if (clientList == null || clientList.Count == 0)
  5874. {
  5875. builder.MoveToCell(0, 1, 0, 0);
  5876. builder.Write("【本团尚未录入客户名单,请联系主管!】");
  5877. }
  5878. else
  5879. {
  5880. for (int i = 0; i < clientList.Count; i++)
  5881. {
  5882. builder.MoveToCell(0, i + 1, 0, 0);
  5883. builder.Write((i + 1).ToString());
  5884. builder.MoveToCell(0, i + 1, 1, 0);
  5885. builder.Write(clientList[i].LastName.ToString() + clientList[i].FirstName.ToString());
  5886. builder.MoveToCell(0, i + 1, 2, 0);
  5887. builder.Write(clientList[i].Sex.ToString());
  5888. builder.MoveToCell(0, i + 1, 3, 0);
  5889. builder.Write(clientList[i].BirthDay);
  5890. builder.MoveToCell(0, i + 1, 4, 0);
  5891. builder.Write(clientList[i].CompanyFullName + clientList[i].Job.ToString());
  5892. }
  5893. }
  5894. while (tableClient.Rows.Count > clientList.Count + 1)
  5895. {
  5896. tableClient.Rows.RemoveAt(clientList.Count + 1);
  5897. }
  5898. #endregion
  5899. #region 商邀行程
  5900. string content = "", temp1 = "", temp2 = "";
  5901. //获取数据,放到datatable
  5902. var listapt = await _sqlSugar.Queryable<Grp_ApprovalTravel>()
  5903. .LeftJoin<Grp_ApprovalTravelDetails>((at, atd) => at.Id == atd.ParentId)
  5904. .Where((at, atd) => at.IsDel == 0 && at.Diid == dto.DiId)
  5905. .Select((at, atd) => new
  5906. {
  5907. at.Id,
  5908. at.Date,
  5909. atd.Time,
  5910. atd.Details
  5911. })
  5912. .ToListAsync();
  5913. if (listapt == null || listapt.Count == 0)
  5914. {
  5915. content = "尚未生成商邀行程";
  5916. }
  5917. else
  5918. {
  5919. for (int i = 0; i < listapt.Count; i++)
  5920. {
  5921. content = content + " " + listapt[i].Date + "\r\n";
  5922. temp1 = listapt[i].Time;
  5923. temp2 = listapt[i].Details;
  5924. if (!string.IsNullOrEmpty(temp1))
  5925. {
  5926. string[] str = temp1.Replace("\r\n", " ").Split(new char[] { ' ' }, StringSplitOptions.RemoveEmptyEntries);
  5927. string[] str2 = temp2.Replace("\r\n", " ").Split(new char[] { ' ' }, StringSplitOptions.RemoveEmptyEntries);
  5928. for (int j = 0; j < str.Length; j++)
  5929. {
  5930. content = content + " " + str[j] + str2[j] + "\r\n";
  5931. }
  5932. }
  5933. }
  5934. }
  5935. if (doc.Range.Bookmarks["Content"] != null)
  5936. {
  5937. Bookmark mark = doc.Range.Bookmarks["Content"];
  5938. mark.Text = content;
  5939. }
  5940. while (tableClient.Rows.Count > 1 + clientList.Count)
  5941. {
  5942. tableClient.Rows.RemoveAt(1 + clientList.Count);
  5943. }
  5944. #endregion
  5945. #endregion
  5946. string strFileName = $"/AskData/省外办出访请示{Guid.NewGuid().ToString()}.doc";
  5947. doc.Save(AppSettingsHelper.Get("WordBasePath") + strFileName); //"C:\\Server\\File\\OA2023\\Office\\Word" + strFileName
  5948. var url = AppSettingsHelper.Get("WordBaseUrl") + AppSettingsHelper.Get("WordFtpPath") + strFileName; //"C:\\Server\\File\\OA2023\\Office\\Word" + strFileName
  5949. return Ok(JsonView(true, "操作成功!", new { url = url }));
  5950. }
  5951. /// <summary>
  5952. /// 请示资料信息
  5953. /// 市外办下载
  5954. /// </summary>
  5955. /// <param name="dto"></param>
  5956. /// <returns></returns>
  5957. [HttpPost]
  5958. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  5959. public async Task<IActionResult> AskDataCityFAOPFile(AskDataProvinceFileDto dto)
  5960. {
  5961. if (dto.DataIdItem.Length < 1) return Ok(JsonView(false, $"请传入有效的数据Id集合!"));
  5962. if (dto.DiId < 1) return Ok(JsonView(false, $"请传入有效的数据DiId参数!"));
  5963. //大写数字序号
  5964. string[] num = { "一", "二", "三", "四", "五", "六", "七", "八", "九", "十" ,
  5965. "十一","十二","十三","十四","十五","十六","十七","十八","十九","二十",
  5966. "二十一","二十二","二十三","二十四","二十五","二十六","二十七","二十八","二十九","三十",
  5967. "三十一","三十二","三十三","三十四","三十五","三十六","三十七","三十八","三十九","四十"};
  5968. //数据源
  5969. var idItem = dto.DataIdItem.OrderBy(x => x);
  5970. var askDatas = await _sqlSugar.Queryable<Res_AskData>().Where(x => x.IsDel == 0 && idItem.Contains(x.Id)).ToListAsync();
  5971. //书签名
  5972. string Area = "",
  5973. Official = "";
  5974. //表格数据源
  5975. DataTable dt = new DataTable();
  5976. dt.Columns.Add("Country", typeof(string));
  5977. dt.Columns.Add("TalkCase", typeof(string));
  5978. int id = 0;
  5979. int TalkcaseOrder = 0;
  5980. int RowOrder = 0;
  5981. //填充数据
  5982. for (int i = 0; i < askDatas.Count; i++)
  5983. {
  5984. var ad = askDatas[i];
  5985. Area = Area + ad.Country + ad.Area + "、";
  5986. Official = Official + ad.UnitName + "、";
  5987. //填充表格数据源
  5988. DataRow dr = dt.NewRow();
  5989. if (i > 0)
  5990. {
  5991. var adtemp = askDatas[i - 1];
  5992. if (ad.Country == adtemp.Country)
  5993. {
  5994. TalkcaseOrder++;
  5995. dt.Rows[RowOrder]["TalkCase"] = dt.Rows[RowOrder]["TalkCase"] + "\r\n" + (1 + TalkcaseOrder).ToString() + "、" + ad.TalkCase;
  5996. }
  5997. else
  5998. {
  5999. RowOrder++;
  6000. dr = dt.NewRow();
  6001. dr["Country"] = "(" + num[RowOrder] + ")" + ad.Country;
  6002. dr["TalkCase"] = "1、" + ad.TalkCase;
  6003. dt.Rows.Add(dr);
  6004. TalkcaseOrder = 0;
  6005. }
  6006. }
  6007. else
  6008. {
  6009. dr["Country"] = "(" + num[i] + ")" + ad.Country;
  6010. dr["TalkCase"] = "1、" + ad.TalkCase;
  6011. dt.Rows.Add(dr);
  6012. }
  6013. id = ad.Id;
  6014. }
  6015. //模板路径
  6016. var tempPath = AppSettingsHelper.Get("WordBasePath") + "Template/市外办出访请示.doc";
  6017. //载入模板
  6018. Aspose.Words.Document doc = new Aspose.Words.Document(tempPath);
  6019. #region 填充word内容
  6020. #region 填充请示内容
  6021. //利用DocumentBuilder处理表格或是书签
  6022. Aspose.Words.DocumentBuilder builder = new DocumentBuilder(doc);
  6023. string bookmark = "";
  6024. try
  6025. {
  6026. for (int i = 0; i < dt.Rows.Count; i++)
  6027. {
  6028. bookmark = bookmark
  6029. + dt.Rows[i]["Country"].ToString() + "\r\n"
  6030. + dt.Rows[i]["TalkCase"].ToString() + "\r\n";
  6031. }
  6032. }
  6033. catch
  6034. {
  6035. }
  6036. #endregion
  6037. #region 填充word模板书签内容
  6038. //团组信息
  6039. var di = await _sqlSugar.Queryable<Grp_DelegationInfo>().Where(x => x.IsDel == 0 && x.Id == dto.DiId).FirstAsync();
  6040. //接团客户名单实例
  6041. string client_sql = string.Format(@"
  6042. SELECT
  6043. tcl.Id,
  6044. tcl.DiId,
  6045. tcl.IsAccompany,
  6046. temp.*,
  6047. u.CnName AS Operator,
  6048. tcl.CreateTime AS OperatingTime
  6049. FROM
  6050. Grp_TourClientList tcl
  6051. LEFT JOIN (
  6052. SELECT
  6053. dc.Id AS DcId,
  6054. dc.LastName,
  6055. dc.FirstName,
  6056. ccom.CompanyFullName,
  6057. dc.Job,
  6058. cc.CertNo AS IDCardNo,
  6059. dc.Sex,
  6060. dc.BirthDay
  6061. FROM
  6062. Crm_DeleClient dc
  6063. LEFT JOIN Crm_CustomerCompany ccom ON dc.CrmCompanyId = ccom.Id
  6064. AND ccom.IsDel = 0
  6065. LEFT JOIN Crm_CustomerCert cc ON dc.Id = cc.DcId
  6066. AND cc.SdId = 773
  6067. AND cc.IsDel = 0
  6068. WHERE
  6069. dc.IsDel = 0
  6070. ) temp ON temp.DcId = tcl.ClientId
  6071. LEFT JOIN Sys_Users u ON tcl.CreateUserId = u.Id
  6072. WHERE
  6073. tcl.IsDel = 0
  6074. AND tcl.DiId = {0}", dto.DiId);
  6075. var clientList = await _sqlSugar.SqlQueryable<TourClientListDetailsView>(client_sql).ToListAsync();
  6076. //机票代码实例
  6077. DataTable airTable = GeneralMethod.GetTableByBlackCode(dto.DiId);
  6078. //三字码
  6079. var threeCodes = await _sqlSugar.Queryable<Res_ThreeCode>().Where(x => x.IsDel == 0).ToListAsync();
  6080. //团名
  6081. if (doc.Range.Bookmarks["TeamName"] != null)
  6082. {
  6083. Bookmark mark = doc.Range.Bookmarks["TeamName"];
  6084. mark.Text = di.TeamName;
  6085. }
  6086. //团长名
  6087. if (doc.Range.Bookmarks["ClientName2"] != null)
  6088. {
  6089. Bookmark mark = doc.Range.Bookmarks["ClientName2"];
  6090. if (clientList == null || clientList.Count == 0)
  6091. {
  6092. mark.Text = "【本团尚未录入客户名单,请联系主管!】";
  6093. }
  6094. else
  6095. {
  6096. mark.Text = clientList[0].CompanyFullName + clientList[0].Job + clientList[0].LastName + clientList[0].FirstName;
  6097. }
  6098. }
  6099. if (doc.Range.Bookmarks["ClientName"] != null)
  6100. {
  6101. Bookmark mark = doc.Range.Bookmarks["ClientName"];
  6102. if (clientList == null || clientList.Count == 0)
  6103. {
  6104. mark.Text = "【本团尚未录入客户名单,请联系主管!】";
  6105. }
  6106. else
  6107. {
  6108. mark.Text = clientList[0].CompanyFullName + clientList[0].Job + clientList[0].LastName + clientList[0].FirstName;
  6109. }
  6110. }
  6111. //跟团人数
  6112. if (doc.Range.Bookmarks["ClientNumber2"] != null)
  6113. {
  6114. Bookmark mark = doc.Range.Bookmarks["ClientNumber2"];
  6115. mark.Text = di.VisitPNumber.ToString();
  6116. }
  6117. if (doc.Range.Bookmarks["ClientNumber"] != null)
  6118. {
  6119. Bookmark mark = doc.Range.Bookmarks["ClientNumber"];
  6120. mark.Text = di.VisitPNumber.ToString();
  6121. }
  6122. //地区
  6123. if (doc.Range.Bookmarks["Area"] != null)
  6124. {
  6125. Bookmark mark = doc.Range.Bookmarks["Area"];
  6126. mark.Text = Area.TrimEnd('、');
  6127. }
  6128. if (doc.Range.Bookmarks["Area2"] != null)
  6129. {
  6130. Bookmark mark = doc.Range.Bookmarks["Area2"];
  6131. mark.Text = Area.TrimEnd('、');
  6132. }
  6133. //出访目的
  6134. if (doc.Range.Bookmarks["Destination2"] != null)
  6135. {
  6136. Bookmark mark = doc.Range.Bookmarks["Destination2"];
  6137. mark.Text = di.VisitPurpose;
  6138. }
  6139. if (doc.Range.Bookmarks["Destination"] != null)
  6140. {
  6141. Bookmark mark = doc.Range.Bookmarks["Destination"];
  6142. mark.Text = di.VisitPurpose;
  6143. }
  6144. //邀请方
  6145. if (doc.Range.Bookmarks["Official"] != null)
  6146. {
  6147. Bookmark mark = doc.Range.Bookmarks["Official"];
  6148. mark.Text = Official.TrimEnd('、');
  6149. }
  6150. //出访日期
  6151. if (doc.Range.Bookmarks["VisitDate"] != null)
  6152. {
  6153. Bookmark mark = doc.Range.Bookmarks["VisitDate"];
  6154. mark.Text = di.VisitStartDate.Year + "年"
  6155. + di.VisitStartDate.Month + "月"
  6156. + di.VisitStartDate.Day + "日"
  6157. + "至"
  6158. + di.VisitEndDate.Month + "月"
  6159. + di.VisitEndDate.Day + "日";
  6160. }
  6161. //请示范例
  6162. if (doc.Range.Bookmarks["Temp"] != null)
  6163. {
  6164. Bookmark mark = doc.Range.Bookmarks["Temp"];
  6165. mark.Text = bookmark;
  6166. }
  6167. //出访起止日期
  6168. if (doc.Range.Bookmarks["Duration"] != null)
  6169. {
  6170. Bookmark mark = doc.Range.Bookmarks["Duration"];
  6171. mark.Text = di.VisitStartDate.Year + "年"
  6172. + di.VisitStartDate.Month + "月"
  6173. + di.VisitStartDate.Day + "日"
  6174. + "至"
  6175. + di.VisitEndDate.Month + "月"
  6176. + di.VisitEndDate.Day + "日";
  6177. }
  6178. //出访天数
  6179. if (doc.Range.Bookmarks["Durationdays"] != null)
  6180. {
  6181. Bookmark mark = doc.Range.Bookmarks["Durationdays"];
  6182. mark.Text = di.VisitDays.ToString();
  6183. }
  6184. if (doc.Range.Bookmarks["Durationdays2"] != null)
  6185. {
  6186. Bookmark mark = doc.Range.Bookmarks["Durationdays2"];
  6187. mark.Text = di.VisitDays.ToString();
  6188. }
  6189. #region 根据黑屏代码获取国家呆的天数;获取路线
  6190. string result_CountryAndDay = "", result_Air = "";
  6191. if (airTable.Rows[airTable.Rows.Count - 1]["Day"].ToString() == "")
  6192. {
  6193. result_CountryAndDay = airTable.Rows[airTable.Rows.Count - 1]["Fliagtcode"].ToString();
  6194. result_Air = result_CountryAndDay;
  6195. }
  6196. else
  6197. {
  6198. string place_start, place_end;
  6199. DateTime time_start, time_end;
  6200. Res_ThreeCode three = null;
  6201. for (int i = 0; i < airTable.Rows.Count; i++)
  6202. {
  6203. //三字码实例
  6204. three = threeCodes.Find(x => x.Three == airTable.Rows[i]["Three"].ToString().Substring(0, 3));
  6205. if (three == null)
  6206. {
  6207. place_start = "【未收录三字码" + airTable.Rows[i]["Three"].ToString().Substring(0, 3) + ",请联系机票同事】";
  6208. }
  6209. else
  6210. {
  6211. if (string.IsNullOrEmpty(three.Country))
  6212. {
  6213. place_start = "【本三字码" + airTable.Rows[i]["Three"].ToString().Substring(0, 3) + "未录入国家信息,请联系机票同事】";
  6214. }
  6215. else
  6216. {
  6217. place_start = three.City;
  6218. }
  6219. }
  6220. three = threeCodes.Find(x => x.Three == airTable.Rows[i]["Three"].ToString().Substring(3, 3));
  6221. if (three == null)
  6222. {
  6223. place_end = "【未收录" + airTable.Rows[i]["Three"].ToString().Substring(0, 3) + ",请联系机票同事】";
  6224. }
  6225. else
  6226. {
  6227. if (string.IsNullOrEmpty(three.Country))
  6228. {
  6229. place_end = "【本三字码" + airTable.Rows[i]["Three"].ToString().Substring(0, 3) + "未录入国家信息,请联系机票同事】";
  6230. }
  6231. else
  6232. {
  6233. place_end = three.City;
  6234. }
  6235. }
  6236. result_Air += place_start + "一" + place_end + "一";
  6237. //获取国家和呆的天数
  6238. if (i > 0 && i < airTable.Rows.Count - 1)
  6239. {
  6240. time_start = Convert.ToDateTime(airTable.Rows[i - 1]["Day"]);
  6241. time_end = Convert.ToDateTime(airTable.Rows[i]["Day"]);
  6242. TimeSpan daycount = time_end.Subtract(time_start);
  6243. result_CountryAndDay += place_start + daycount.Days + "天" + ",";
  6244. }
  6245. }
  6246. }
  6247. #endregion
  6248. //每个国家呆的天数
  6249. if (doc.Range.Bookmarks["CountryAndDay"] != null)
  6250. {
  6251. Bookmark mark = doc.Range.Bookmarks["CountryAndDay"];
  6252. mark.Text = result_CountryAndDay.TrimEnd(',');
  6253. }
  6254. #region 出访路线城市字符串去重
  6255. result_Air = result_Air.TrimEnd('一');
  6256. string[] airTemp = result_Air.Split('一');
  6257. airTemp = airTemp.Distinct().ToArray();
  6258. result_Air = "";
  6259. foreach (string str in airTemp)
  6260. {
  6261. result_Air += str + "一";
  6262. }
  6263. result_Air = result_Air + airTemp[0];
  6264. #endregion
  6265. //出访路线
  6266. if (doc.Range.Bookmarks["Air"] != null)
  6267. {
  6268. Bookmark mark = doc.Range.Bookmarks["Air"];
  6269. mark.Text = result_Air;
  6270. }
  6271. #endregion
  6272. #region 客户名单
  6273. NodeCollection allTables = doc.GetChildNodes(NodeType.Table, true);
  6274. Aspose.Words.Tables.Table tableClient = allTables[0] as Aspose.Words.Tables.Table;
  6275. if (clientList == null || clientList.Count == 0)
  6276. {
  6277. builder.MoveToCell(0, 1, 0, 0);
  6278. builder.Write("【本团尚未录入客户名单,请联系主管!】");
  6279. }
  6280. else
  6281. {
  6282. for (int i = 0; i < clientList.Count; i++)
  6283. {
  6284. builder.MoveToCell(0, i + 1, 0, 0);
  6285. builder.Write((i + 1).ToString());
  6286. builder.MoveToCell(0, i + 1, 1, 0);
  6287. builder.Write(clientList[i].LastName.ToString() + clientList[i].FirstName.ToString());
  6288. builder.MoveToCell(0, i + 1, 2, 0);
  6289. builder.Write(clientList[i].Sex.ToString());
  6290. builder.MoveToCell(0, i + 1, 3, 0);
  6291. builder.Write(clientList[i].BirthDay);
  6292. builder.MoveToCell(0, i + 1, 4, 0);
  6293. builder.Write(clientList[i].CompanyFullName + clientList[i].Job.ToString());
  6294. }
  6295. }
  6296. while (tableClient.Rows.Count > clientList.Count + 1)
  6297. {
  6298. tableClient.Rows.RemoveAt(clientList.Count + 1);
  6299. }
  6300. #endregion
  6301. #region 商邀行程
  6302. string content = "", temp1 = "", temp2 = "";
  6303. //获取数据,放到datatable
  6304. var listapt = await _sqlSugar.Queryable<Grp_ApprovalTravel>()
  6305. .LeftJoin<Grp_ApprovalTravelDetails>((at, atd) => at.Id == atd.ParentId)
  6306. .Where((at, atd) => at.IsDel == 0 && at.Diid == dto.DiId)
  6307. .Select((at, atd) => new
  6308. {
  6309. at.Id,
  6310. at.Date,
  6311. atd.Time,
  6312. atd.Details
  6313. })
  6314. .ToListAsync();
  6315. if (listapt == null || listapt.Count == 0)
  6316. {
  6317. content = "尚未生成商邀行程";
  6318. }
  6319. else
  6320. {
  6321. for (int i = 0; i < listapt.Count; i++)
  6322. {
  6323. content = content + " " + listapt[i].Date + "\r\n";
  6324. temp1 = listapt[i].Time;
  6325. temp2 = listapt[i].Details;
  6326. if (!string.IsNullOrEmpty(temp1))
  6327. {
  6328. string[] str = temp1.Replace("\r\n", " ").Split(new char[] { ' ' }, StringSplitOptions.RemoveEmptyEntries);
  6329. string[] str2 = temp2.Replace("\r\n", " ").Split(new char[] { ' ' }, StringSplitOptions.RemoveEmptyEntries);
  6330. for (int j = 0; j < str.Length; j++)
  6331. {
  6332. content = content + " " + str[j] + str2[j] + "\r\n";
  6333. }
  6334. }
  6335. }
  6336. }
  6337. if (doc.Range.Bookmarks["Content"] != null)
  6338. {
  6339. Bookmark mark = doc.Range.Bookmarks["Content"];
  6340. mark.Text = content;
  6341. }
  6342. while (tableClient.Rows.Count > 1 + clientList.Count)
  6343. {
  6344. tableClient.Rows.RemoveAt(1 + clientList.Count);
  6345. }
  6346. #endregion
  6347. #endregion
  6348. string strFileName = $"/AskData/市外办出访请示{Guid.NewGuid().ToString()}.doc";
  6349. doc.Save(AppSettingsHelper.Get("WordBasePath") + strFileName); //"C:\\Server\\File\\OA2023\\Office\\Word" + strFileName
  6350. var url = AppSettingsHelper.Get("WordBaseUrl") + AppSettingsHelper.Get("WordFtpPath") + strFileName; //"C:\\Server\\File\\OA2023\\Office\\Word" + strFileName
  6351. return Ok(JsonView(true, "操作成功!", new { url = url }));
  6352. }
  6353. #endregion
  6354. #region 机票黑屏代码
  6355. /// <summary>
  6356. /// 根据团组Id查询黑屏代码列表
  6357. /// </summary>
  6358. /// <param name="dto"></param>
  6359. /// <returns></returns>
  6360. [HttpPost]
  6361. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6362. public async Task<IActionResult> QueryTicketBlackCodeByDiId(QueryTicketBlackCodeByDiIdDto dto)
  6363. {
  6364. Result groupData = await _ticketBlackCodeRep.QueryTicketBlackCodeByDiId(dto);
  6365. if (groupData.Code != 0)
  6366. {
  6367. return Ok(JsonView(false, groupData.Msg));
  6368. }
  6369. return Ok(JsonView(true, groupData.Msg, groupData.Data));
  6370. }
  6371. /// <summary>
  6372. /// 根据黑屏代码数据Id查询信息
  6373. /// </summary>
  6374. /// <param name="dto"></param>
  6375. /// <returns></returns>
  6376. [HttpPost]
  6377. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6378. public async Task<IActionResult> QueryTicketBlackCodeById(QueryTicketBlackCodeByIdDto dto)
  6379. {
  6380. Result groupData = await _ticketBlackCodeRep.QueryTicketBlackCodeById(dto);
  6381. if (groupData.Code != 0)
  6382. {
  6383. return Ok(JsonView(false, groupData.Msg));
  6384. }
  6385. return Ok(JsonView(true, groupData.Msg, groupData.Data));
  6386. }
  6387. /// <summary>
  6388. /// 黑屏代码操作(Status:1.新增,2.修改)
  6389. /// </summary>
  6390. /// <param name="dto"></param>
  6391. /// <returns></returns>
  6392. [HttpPost]
  6393. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6394. public async Task<IActionResult> OpTicketBlackCode(OpTicketBlackCodeDto dto)
  6395. {
  6396. Result groupData = await _ticketBlackCodeRep.OpTicketBlackCode(dto);
  6397. if (groupData.Code != 0)
  6398. {
  6399. return Ok(JsonView(false, groupData.Msg));
  6400. }
  6401. try
  6402. {
  6403. var tongzhi = _ticketBlackCodeRep.DescBlackToVisa(dto.DiId);
  6404. if (tongzhi.Code == 0)
  6405. {
  6406. var info = tongzhi.Data.GetType().GetProperty("info").GetValue(tongzhi.Data) as List<string>;
  6407. var data = tongzhi.Data.GetType().GetProperty("data").GetValue(tongzhi.Data) as List<CountryDataTime>;
  6408. if (data != null)
  6409. await AppNoticeLibrary.SendUserMsg_blackAirInfo_ToVisaUser(dto.DiId, data);
  6410. }
  6411. //行程代码变更通知
  6412. await AppNoticeLibrary.SendUserMsg_GroupShare_ToDP(dto.DiId, dto.CreateUserId);
  6413. }
  6414. catch (Exception)
  6415. {
  6416. throw;
  6417. }
  6418. return Ok(JsonView(true, groupData.Msg, groupData.Data));
  6419. }
  6420. /// <summary>
  6421. /// 删除黑屏代码
  6422. /// </summary>
  6423. /// <param name="dto"></param>
  6424. /// <returns></returns>
  6425. [HttpPost]
  6426. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6427. public async Task<IActionResult> DelTicketBlackCode(DelBaseDto dto)
  6428. {
  6429. var res = await _ticketBlackCodeRep.SoftDeleteByIdAsync<Air_TicketBlackCode>(dto.Id.ToString(), dto.DeleteUserId);
  6430. if (!res)
  6431. {
  6432. return Ok(JsonView(false, "删除失败"));
  6433. }
  6434. return Ok(JsonView(true, "删除成功!"));
  6435. }
  6436. #endregion
  6437. #region 翻译人员
  6438. /// <summary>
  6439. /// 翻译人员库
  6440. /// Init
  6441. /// </summary>
  6442. /// <returns></returns>
  6443. [HttpGet]
  6444. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6445. public async Task<IActionResult> TranslatorLibraryInit()
  6446. {
  6447. var currencyData = await _sqlSugar.Queryable<Sys_SetData>()
  6448. .Where(x => x.IsDel == 0 && x.STid == 66)
  6449. .Select(x => new
  6450. {
  6451. x.Id,
  6452. x.Name,
  6453. })
  6454. .ToListAsync();
  6455. var officialDutyData = await _sqlSugar.Queryable<Res_OfficialActivities>()
  6456. .LeftJoin<Grp_DelegationInfo>((oa, di) => oa.DiId == di.Id)
  6457. .Where((oa, di) => oa.IsDel == 0)
  6458. .Select((oa, di) => new
  6459. {
  6460. oa.Id,
  6461. oa.Client,
  6462. oa.DiId,
  6463. di.TeamName
  6464. })
  6465. .ToListAsync();
  6466. var officialDutyData1 = officialDutyData.GroupBy(x => x.DiId)
  6467. .Select(g => new
  6468. {
  6469. id = g.Key,
  6470. label = g.FirstOrDefault()?.TeamName ?? "-",
  6471. children = g.Select(g1 => new { id = g1.Id, label = g1.Client }).ToList()
  6472. })
  6473. .ToList();
  6474. var view = new
  6475. {
  6476. currencyData,
  6477. officialDutyData = officialDutyData1
  6478. };
  6479. return Ok(JsonView(view));
  6480. }
  6481. /// <summary>
  6482. /// 翻译人员库
  6483. /// 详情
  6484. /// </summary>
  6485. /// <param name="id"></param>
  6486. /// <returns></returns>
  6487. [HttpGet("id")]
  6488. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6489. public async Task<IActionResult> TranslatorLibraryInfo(int id)
  6490. {
  6491. if (id < 1) return Ok(JsonView(false, "请传入有效的Id!"));
  6492. var info = await _translatorRep.Query(x => x.Id == id).FirstAsync();
  6493. EncryptionProcessor.DecryptProperties(info);
  6494. var view = _mapper.Map<TranslatorLibraryInfoView>(info);
  6495. //var data1 = await _sqlSugar.Queryable<Grp_OfficialDutyLinkTranslator>().Where(x => x.TranslatorId == id).Select(x => x.Id).ToArrayAsync();
  6496. view.OfficialDutyIdItem = await _sqlSugar
  6497. .Queryable<Grp_OfficialDutyLinkTranslator>()
  6498. .Where(x => x.IsDel == 0 && x.TranslatorId == id)
  6499. .Select(x => x.OfficialDutyId)
  6500. .ToArrayAsync();
  6501. return Ok(JsonView(view));
  6502. }
  6503. /// <summary>
  6504. /// 翻译人员库
  6505. /// Item
  6506. /// </summary>
  6507. /// <param name="dto"></param>
  6508. /// <returns></returns>
  6509. [HttpPost]
  6510. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6511. public async Task<IActionResult> TranslatorLibraryItem(TranslatorLibraryItemDto dto)
  6512. {
  6513. if (dto.PortType < 1 || dto.PortType > 3) return Ok(JsonView(false, MsgTips.Port));
  6514. var name = AesEncryptionHelper.Encrypt(dto.Name);
  6515. RefAsync<int> total = 0;
  6516. var view = await _sqlSugar.Queryable<Res_TranslatorLibrary>()
  6517. .Where(x => x.IsDel == 0)
  6518. .WhereIF(!string.IsNullOrEmpty(name), x => x.Name.Contains(name))
  6519. .Select(x => new TranslatorLibraryItemView()
  6520. {
  6521. Id = x.Id,
  6522. Area = x.Area,
  6523. Name = x.Name,
  6524. Sex = x.Sex == 0 ? "未设置" : x.Sex == 1 ? "男" : "女",
  6525. Tel = x.Tel,
  6526. Email = x.Email,
  6527. WechatNo = x.WechatNo,
  6528. Language = x.Language,
  6529. Price = x.Price,
  6530. Currency = SqlFunc.Subqueryable<Sys_SetData>().Where(x1 => x1.Id == x.Currency).Select(x1 => x1.Name) ?? "-",
  6531. CreateUserName = SqlFunc.Subqueryable<Sys_Users>().Where(x1 => x1.Id == x.CreateUserId).Select(x1 => x1.CnName) ?? "-",
  6532. LastUpddateTime = x.LastUpdateTime,
  6533. CreateTime = x.CreateTime,
  6534. Remark = x.Remark
  6535. })
  6536. .OrderByDescending(x => x.CreateTime)
  6537. .ToPageListAsync(dto.PageIndex, dto.PageSize, total);
  6538. foreach (var item in view) EncryptionProcessor.DecryptProperties(item);
  6539. return Ok(JsonView(view, total));
  6540. }
  6541. /// <summary>
  6542. /// 翻译人员库
  6543. /// OP
  6544. /// </summary>
  6545. /// <param name="dto"></param>
  6546. /// <returns></returns>
  6547. [HttpPost]
  6548. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6549. public async Task<IActionResult> TranslatorLibraryOp([FromForm] TranslatorLibraryOpDto dto)
  6550. {
  6551. if (dto.PortType < 1 || dto.PortType > 3) return Ok(JsonView(false, MsgTips.Port));
  6552. if (dto.Status < 1 || dto.Status > 2) return Ok(JsonView(false, MsgTips.Status));
  6553. var currUserInfo = JwtHelper.SerializeJwt(HttpContext.Request.Headers.Authorization);
  6554. if (currUserInfo == null) return Ok(JsonView(false, "请传入token!"));
  6555. var fileNames = new List<string>();
  6556. //D:\FTP\File\OA2023\Office\GrpFile\TranslatorLibrary
  6557. var fileBasePath = $"{AppSettingsHelper.Get("GrpFileBasePath")}TranslatorLibrary";
  6558. #region 保存文件
  6559. if (dto.Files != null)
  6560. {
  6561. if (!System.IO.Directory.Exists(fileBasePath))
  6562. {
  6563. System.IO.Directory.CreateDirectory(fileBasePath);
  6564. }
  6565. foreach (var item in dto.Files)
  6566. {
  6567. if (item.Length < 1) continue;
  6568. var file = item;
  6569. // 将文件保存到指定位置
  6570. var filePath = Path.Combine(fileBasePath, file.FileName);
  6571. using (var stream = new FileStream(filePath, FileMode.Create))
  6572. {
  6573. await file.CopyToAsync(stream);
  6574. }
  6575. //验证文件是否上传成功
  6576. if (!System.IO.File.Exists(filePath))
  6577. {
  6578. foreach (var filePathStr in fileNames)
  6579. {
  6580. System.IO.File.Delete(Path.Combine(fileBasePath, filePathStr));
  6581. }
  6582. return Ok(JsonView(false, "文件上传失败!"));
  6583. }
  6584. fileNames.Add(file.FileName);
  6585. }
  6586. }
  6587. #endregion
  6588. #region 参数处理
  6589. EncryptionProcessor.EncryptProperties(dto);
  6590. var translatorInfo = _mapper.Map<Res_TranslatorLibrary>(dto);
  6591. translatorInfo.LastUpdateTime = translatorInfo.CreateTime;
  6592. translatorInfo.CreateUserId = currUserInfo.UserId;
  6593. #endregion
  6594. _sqlSugar.BeginTran();
  6595. if (dto.Status == 1)
  6596. {
  6597. //验证重复
  6598. var info = await _translatorRep.Query(x => x.Area.Equals(dto.Area) && x.Name.Equals(dto.Name)).FirstAsync();
  6599. if (info != null) return Ok(JsonView(false, "该条数据已存在!"));
  6600. if (fileNames.Count > 0)
  6601. {
  6602. translatorInfo.Files = AesEncryptionHelper.Encrypt(JsonConvert.SerializeObject(fileNames));
  6603. }
  6604. var id = await _translatorRep.AddAsync(translatorInfo);
  6605. if (id > 0)
  6606. {
  6607. #region 新增(公务信息关联翻译人员) 关联信息
  6608. if (dto.officialDutyIdItem != null && dto.officialDutyIdItem.Length > 0)
  6609. {
  6610. var officialDutyLinkTranslators = new List<Grp_OfficialDutyLinkTranslator>();
  6611. foreach (var officialDutyId in dto.officialDutyIdItem)
  6612. {
  6613. officialDutyLinkTranslators.Add(new Grp_OfficialDutyLinkTranslator()
  6614. {
  6615. TranslatorId = id,
  6616. OfficialDutyId = officialDutyId,
  6617. CreateUserId = currUserInfo.UserId,
  6618. Remark = $"翻译人员库-->添加"
  6619. });
  6620. }
  6621. if (officialDutyLinkTranslators.Count > 0)
  6622. {
  6623. await _sqlSugar.Insertable(officialDutyLinkTranslators).ExecuteCommandAsync();
  6624. }
  6625. }
  6626. #endregion
  6627. _sqlSugar.CommitTran();
  6628. return Ok(JsonView(true));
  6629. }
  6630. }
  6631. else if (dto.Status == 2)
  6632. {
  6633. var info = await _translatorRep.Query(x => x.Id == dto.Id).FirstAsync();
  6634. if (info != null)
  6635. {
  6636. if (!string.IsNullOrEmpty(info.Files))
  6637. {
  6638. try
  6639. {
  6640. var selectFiles = JsonConvert.DeserializeObject<List<string>>(info.Files);
  6641. fileNames.AddRange(selectFiles);
  6642. }
  6643. catch (Exception)
  6644. {
  6645. }
  6646. }
  6647. }
  6648. if (fileNames.Count > 0)
  6649. {
  6650. translatorInfo.Files = AesEncryptionHelper.Encrypt(JsonConvert.SerializeObject(fileNames));
  6651. }
  6652. translatorInfo.LastUpdateTime = DateTime.Now;
  6653. var upd = await _translatorRep
  6654. .UpdateAsync(x => x.Id == dto.Id, x => new Res_TranslatorLibrary()
  6655. {
  6656. Area = translatorInfo.Area,
  6657. Name = translatorInfo.Name,
  6658. Sex = translatorInfo.Sex,
  6659. Photo = translatorInfo.Photo,
  6660. Tel = translatorInfo.Tel,
  6661. Email = translatorInfo.Email,
  6662. WechatNo = translatorInfo.WechatNo,
  6663. OtherSocialAccounts = translatorInfo.OtherSocialAccounts,
  6664. Language = translatorInfo.Language,
  6665. Price = translatorInfo.Price,
  6666. Currency = translatorInfo.Currency,
  6667. Remark = translatorInfo.Remark,
  6668. Files = translatorInfo.Files,
  6669. LastUpdateTime = translatorInfo.LastUpdateTime,
  6670. LastUpdateUserId = translatorInfo.LastUpdateUserId,
  6671. });
  6672. if (upd)
  6673. {
  6674. #region 更新(公务信息关联翻译人员) 关联信息
  6675. var officialDutyLinkTranslators_select = await _sqlSugar.Queryable<Grp_OfficialDutyLinkTranslator>()
  6676. .Where(x => x.IsDel == 0 && x.TranslatorId == dto.Id)
  6677. .ToListAsync();
  6678. //删除
  6679. var officialDutyLinkTranslatorIds = officialDutyLinkTranslators_select.Select(x => x.Id).ToList();
  6680. if (officialDutyLinkTranslatorIds.Count > 0)
  6681. {
  6682. await _sqlSugar.Updateable<Grp_OfficialDutyLinkTranslator>()
  6683. .SetColumnsIF(officialDutyLinkTranslatorIds.Count > 0, x => new Grp_OfficialDutyLinkTranslator()
  6684. {
  6685. DeleteUserId = currUserInfo.UserId,
  6686. DeleteTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  6687. IsDel = 1,
  6688. Remark = $"翻译人员库-->删除"
  6689. })
  6690. .Where(x => officialDutyLinkTranslatorIds.Contains(x.Id))
  6691. .ExecuteCommandAsync();
  6692. }
  6693. if (dto.officialDutyIdItem != null && dto.officialDutyIdItem.Length > 0)
  6694. {
  6695. //添加
  6696. var officialDutyLinkTranslators = new List<Grp_OfficialDutyLinkTranslator>();
  6697. foreach (var officialDutyId in dto.officialDutyIdItem)
  6698. {
  6699. officialDutyLinkTranslators.Add(new Grp_OfficialDutyLinkTranslator()
  6700. {
  6701. TranslatorId = dto.Id,
  6702. OfficialDutyId = officialDutyId,
  6703. CreateUserId = currUserInfo.UserId,
  6704. Remark = $"翻译人员库-->更新"
  6705. });
  6706. }
  6707. if (officialDutyLinkTranslators.Count > 0)
  6708. {
  6709. await _sqlSugar.Insertable(officialDutyLinkTranslators).ExecuteCommandAsync();
  6710. }
  6711. }
  6712. #endregion
  6713. _sqlSugar.CommitTran();
  6714. return Ok(JsonView(true));
  6715. }
  6716. }
  6717. _sqlSugar.RollbackTran();
  6718. return Ok(JsonView(false));
  6719. }
  6720. /// <summary>
  6721. /// 翻译人员库
  6722. /// Del
  6723. /// </summary>
  6724. /// <param name="id"></param>
  6725. /// <returns></returns>
  6726. [HttpDelete("id")]
  6727. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6728. public async Task<IActionResult> TranslatorLibraryDel(int id)
  6729. {
  6730. if (id < 1) return Ok(JsonView(false, MsgTips.Id));
  6731. var currUserInfo = JwtHelper.SerializeJwt(HttpContext.Request.Headers.Authorization);
  6732. if (currUserInfo == null) return Ok(JsonView(false, "请传入token!"));
  6733. _sqlSugar.BeginTran();
  6734. var upd = await _translatorRep.SoftDeleteByIdAsync<Res_TranslatorLibrary>(id.ToString(), currUserInfo.UserId);
  6735. if (!upd)
  6736. {
  6737. _sqlSugar.RollbackTran();
  6738. return Ok(JsonView(false));
  6739. }
  6740. #region 删除公务出访
  6741. await _sqlSugar.Updateable<Grp_OfficialDutyLinkTranslator>()
  6742. .SetColumns(x => new Grp_OfficialDutyLinkTranslator()
  6743. {
  6744. DeleteUserId = currUserInfo.UserId,
  6745. DeleteTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  6746. IsDel = 1,
  6747. Remark = $"翻译人员库-->删除"
  6748. })
  6749. .Where(x => x.TranslatorId == id)
  6750. .ExecuteCommandAsync();
  6751. #endregion
  6752. _sqlSugar.CommitTran();
  6753. return Ok(JsonView(true));
  6754. }
  6755. #endregion
  6756. #region 策划部供应商资料
  6757. /// <summary>
  6758. /// 策划部供应商资料
  6759. /// Init
  6760. /// </summary>
  6761. /// <returns></returns>
  6762. [HttpGet]
  6763. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6764. public async Task<IActionResult> MediaSupplierInit()
  6765. {
  6766. return Ok(await _mediaSupplierRep.Init());
  6767. }
  6768. /// <summary>
  6769. /// 策划部供应商资料
  6770. /// Info
  6771. /// </summary>
  6772. /// <param name="dto"></param>
  6773. /// <returns></returns>
  6774. [HttpPost]
  6775. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6776. public async Task<IActionResult> MediaSupplierInfo(MediaSupplierInfoDto dto)
  6777. {
  6778. if (!_portIds.Contains(dto.PortType)) return Ok(JsonView(false, MsgTips.Port));
  6779. if (dto.Id < 1) return Ok(JsonView(false, MsgTips.Port));
  6780. return Ok(await _mediaSupplierRep.Info(dto));
  6781. }
  6782. /// <summary>
  6783. /// 策划部供应商资料
  6784. /// 分页查询
  6785. /// </summary>
  6786. /// <param name="dto"></param>
  6787. /// <returns></returns>
  6788. [HttpPost]
  6789. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6790. public async Task<IActionResult> MediaSupplierPageItem(MediaSupplierPageItemDto dto)
  6791. {
  6792. if (!_portIds.Contains(dto.PortType)) return Ok(JsonView(false, MsgTips.Port));
  6793. return Ok(await _mediaSupplierRep.PageItem(dto));
  6794. }
  6795. /// <summary>
  6796. /// 策划部供应商资料
  6797. /// 操作(添加 Or 编辑)
  6798. /// </summary>
  6799. /// <param name="dto"></param>
  6800. /// <returns></returns>
  6801. [HttpPost]
  6802. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6803. public async Task<IActionResult> MediaSupplierOp(MediaSupplierAddOrEditDto dto)
  6804. {
  6805. var userId = dto.CurrUserId;
  6806. if (!_portIds.Contains(dto.PortType)) return Ok(JsonView(false, MsgTips.Port));
  6807. if (userId < 1) return Ok(JsonView(false, MsgTips.UserId));
  6808. return Ok(await _mediaSupplierRep.AddOrEdit(dto));
  6809. }
  6810. /// <summary>
  6811. /// 策划部供应商资料
  6812. /// 删除
  6813. /// </summary>
  6814. /// <param name="dto"></param>
  6815. /// <returns></returns>
  6816. [HttpPost]
  6817. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6818. public async Task<IActionResult> MediaSupplierSoftDel(MediaSupplierSoftDelDto dto)
  6819. {
  6820. int userId = dto.CurrUserId, id = dto.Id;
  6821. if (!_portIds.Contains(dto.PortType)) return Ok(JsonView(false, MsgTips.Port));
  6822. if (userId < 1) return Ok(JsonView(false, MsgTips.UserId));
  6823. if (id < 1) return Ok(JsonView(false, MsgTips.Id));
  6824. return Ok(await _mediaSupplierRep.SoftDel(dto));
  6825. }
  6826. #endregion
  6827. #region 保险国家基础费用
  6828. /// <summary>
  6829. /// 保险国家基础费用
  6830. /// Info
  6831. /// </summary>
  6832. /// <param name="dto"></param>
  6833. /// <returns></returns>
  6834. [HttpPost]
  6835. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6836. public async Task<IActionResult> InsuranceCostInfo(InsuranceCostInfoDto dto)
  6837. {
  6838. if (!_portIds.Contains(dto.PortType)) return Ok(JsonView(false, MsgTips.Port));
  6839. if (dto.Id < 1) return Ok(JsonView(false, MsgTips.Port));
  6840. var id = dto.Id;
  6841. var info = await _sqlSugar.Queryable<Res_BasicInsuranceCost>()
  6842. .Where(x => x.Id == id && x.IsDel == 0)
  6843. .Select(x => new
  6844. {
  6845. x.Id,
  6846. x.IsSchengen,
  6847. x.CountryName,
  6848. x.Cost,
  6849. x.Remark,
  6850. CreateUserName = SqlFunc.Subqueryable<Sys_Users>().Where(s => s.Id == x.CreateUserId).Select(s => s.CnName),
  6851. x.CreateTime
  6852. })
  6853. .FirstAsync();
  6854. return Ok(JsonView(info));
  6855. }
  6856. /// <summary>
  6857. /// 保险国家基础费用
  6858. /// 分页查询
  6859. /// </summary>
  6860. /// <param name="dto"></param>
  6861. /// <returns></returns>
  6862. [HttpPost]
  6863. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6864. public async Task<IActionResult> InsuranceCostPageItem(InsuranceCostPageItemDto dto)
  6865. {
  6866. if (!_portIds.Contains(dto.PortType)) return Ok(JsonView(false, MsgTips.Port));
  6867. RefAsync<int> total = 0;
  6868. var data = await _sqlSugar.Queryable<Res_BasicInsuranceCost>()
  6869. .LeftJoin<Sys_Users>((bic, u) => bic.CreateUserId == u.Id)
  6870. .Where((bic, u) => bic.IsDel == 0)
  6871. .WhereIF(!string.IsNullOrEmpty(dto.Search), (bic, u) => bic.CountryName.Contains(dto.Search))
  6872. .OrderByDescending((bic, u) => bic.Id)
  6873. .Select((bic, u) => new
  6874. {
  6875. bic.Id,
  6876. bic.IsSchengen,
  6877. bic.CountryName,
  6878. bic.Cost,
  6879. bic.Remark,
  6880. CreateUserName = u.CnName,
  6881. bic.CreateTime
  6882. })
  6883. .ToPageListAsync(dto.PageIndex, dto.PageSize, total);
  6884. return Ok(JsonView(data, total));
  6885. }
  6886. /// <summary>
  6887. /// 保险国家基础费用
  6888. /// 操作(添加 Or 编辑)
  6889. /// </summary>
  6890. /// <param name="dto"></param>
  6891. /// <returns></returns>
  6892. [HttpPost]
  6893. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6894. public async Task<IActionResult> InsuranceCostOp(InsuranceCostOpDto dto)
  6895. {
  6896. var result = new JsonView() { Code = 400, Msg = "操作失败" };
  6897. var userId = dto.CurrUserId;
  6898. if (!_portIds.Contains(dto.PortType)) return Ok(JsonView(false, MsgTips.Port));
  6899. if (userId < 1) return Ok(JsonView(false, MsgTips.UserId));
  6900. var countryName = dto.CountryName.Trim();
  6901. if (string.IsNullOrEmpty(countryName)) return Ok(JsonView(false, $"国家名称不能为空!"));
  6902. var info = _mapper.Map<Res_BasicInsuranceCost>(dto);
  6903. info.CreateUserId = dto.CurrUserId;
  6904. info.CountryName = countryName;
  6905. if (dto.Id < 1) //添加
  6906. {
  6907. var addInfo = await _insuranceCostRep.Query(x => x.CountryName.Equals(countryName)).FirstAsync();
  6908. if (addInfo != null) return Ok(JsonView(false, $"该国家信息已存在,不可添加!"));
  6909. var add = await _insuranceCostRep.AddAsync(info);
  6910. if (add < 1) return Ok(JsonView(false, "添加失败!"));
  6911. }
  6912. else //修改
  6913. {
  6914. var upd = await _insuranceCostRep.UpdateAsync(x => x.Id == dto.Id, x => new Res_BasicInsuranceCost
  6915. {
  6916. IsSchengen = dto.IsSchengen,
  6917. CountryName = dto.CountryName,
  6918. Cost = dto.Cost,
  6919. Remark = dto.Remark
  6920. });
  6921. if (!upd) return Ok(JsonView(false, "修改失败!"));
  6922. }
  6923. return Ok(JsonView(true));
  6924. }
  6925. /// <summary>
  6926. /// 保险国家基础费用
  6927. /// 删除
  6928. /// </summary>
  6929. /// <param name="dto"></param>
  6930. /// <returns></returns>
  6931. [HttpPost]
  6932. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6933. public async Task<IActionResult> InsuranceCostSoftDel(InsuranceCostSoftDelDto dto)
  6934. {
  6935. int userId = dto.CurrUserId, id = dto.Id;
  6936. if (!_portIds.Contains(dto.PortType)) return Ok(JsonView(false, MsgTips.Port));
  6937. if (userId < 1) return Ok(JsonView(false, MsgTips.UserId));
  6938. if (id < 1) return Ok(JsonView(false, MsgTips.Id));
  6939. var del = await _insuranceCostRep.SoftDeleteAsync(x => x.Id == id, userId);
  6940. if (!del) return Ok(JsonView(false));
  6941. return Ok(JsonView(true));
  6942. }
  6943. #endregion
  6944. #region 世运会成本预算明细
  6945. /// <summary>
  6946. /// 世运会成本预算明细
  6947. /// Info
  6948. /// </summary>
  6949. /// <param name="id"></param>
  6950. /// <returns></returns>
  6951. [HttpGet("{id}")]
  6952. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6953. public async Task<IActionResult> GamesBudgetMasterInfo([Required(ErrorMessage = "数据编号不能为空!")] int id)
  6954. {
  6955. return Ok(await _gamesBudgetMasterRep.InfoAsync(id));
  6956. }
  6957. /// <summary>
  6958. /// 世运会成本预算明细
  6959. /// List
  6960. /// </summary>
  6961. /// <param name="dto"></param>
  6962. /// <returns></returns>
  6963. [HttpPost]
  6964. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6965. public async Task<IActionResult> GamesBudgetMasterList(GamesBudgetMasterListDto dto)
  6966. {
  6967. return Ok(await _gamesBudgetMasterRep.ListAsync(dto));
  6968. }
  6969. /// <summary>
  6970. /// 世运会成本预算明细
  6971. /// OP(Add/Edit)
  6972. /// </summary>
  6973. /// <param name="dto"></param>
  6974. /// <returns></returns>
  6975. [HttpPost]
  6976. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  6977. public async Task<IActionResult> GamesBudgetMasterOP(GamesBudgetMasterOPDto dto)
  6978. {
  6979. if (dto.Id > 0)
  6980. {
  6981. //修改
  6982. var editDto = _mapper.Map<GamesBudgetMasterEditDto>(dto);
  6983. return Ok(await _gamesBudgetMasterRep.EditAsync(editDto));
  6984. }
  6985. else if (dto.Id < 1)
  6986. {
  6987. //添加
  6988. var addDto = _mapper.Map<GamesBudgetMasterAddDto>(dto);
  6989. return Ok(await _gamesBudgetMasterRep.AddAsync(addDto));
  6990. }
  6991. return Ok(JsonView(false));
  6992. }
  6993. /// <summary>
  6994. /// 世运会成本预算明细
  6995. /// 删除
  6996. /// </summary>
  6997. /// <param name="dto"></param>
  6998. /// <returns></returns>
  6999. [HttpPost]
  7000. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7001. public async Task<IActionResult> GamesBudgetMasterDel(GamesBudgetMasterDelDto dto)
  7002. {
  7003. return Ok(await _gamesBudgetMasterRep.DelAsync(dto));
  7004. }
  7005. #endregion
  7006. #region 境外用车
  7007. /// <summary>
  7008. /// 境外用车
  7009. /// Init
  7010. /// </summary>
  7011. /// <returns></returns>
  7012. [HttpGet()]
  7013. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7014. public async Task<IActionResult> OverseaVehicleInitAsync()
  7015. {
  7016. return Ok(await _overseaVehicleRep.InitAsync());
  7017. }
  7018. /// <summary>
  7019. /// 境外用车
  7020. /// Info
  7021. /// </summary>
  7022. /// <param name="id"></param>
  7023. /// <returns></returns>
  7024. [HttpGet("{id}")]
  7025. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7026. public async Task<IActionResult> OverseaVehicleInfoAsync([Required(ErrorMessage = "数据编号不能为空!")] int id)
  7027. {
  7028. return Ok(await _overseaVehicleRep.InfoAsync(id));
  7029. }
  7030. /// <summary>
  7031. /// 境外用车
  7032. /// List
  7033. /// </summary>
  7034. /// <param name="dto"></param>
  7035. /// <returns></returns>
  7036. [HttpPost]
  7037. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7038. public async Task<IActionResult> OverseaVehicleListAsync(OverseaVehicleListDto dto)
  7039. {
  7040. return Ok(await _overseaVehicleRep.ListAsync(dto));
  7041. }
  7042. /// <summary>
  7043. /// 境外用车
  7044. /// OP(Add/Edit)
  7045. /// </summary>
  7046. /// <param name="dto"></param>
  7047. /// <returns></returns>
  7048. [HttpPost]
  7049. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7050. public async Task<IActionResult> OverseaVehicleAddOrEditAsync(OverseaVehicleAddOrEditDto dto)
  7051. {
  7052. if (dto.Id > 0)
  7053. {
  7054. //修改
  7055. return Ok(await _overseaVehicleRep.EditAsync(dto));
  7056. }
  7057. else if (dto.Id < 1)
  7058. {
  7059. //添加
  7060. return Ok(await _overseaVehicleRep.AddAsync(dto));
  7061. }
  7062. return Ok(JsonView(false));
  7063. }
  7064. /// <summary>
  7065. /// 境外用车
  7066. /// 删除
  7067. /// </summary>
  7068. /// <param name="dto"></param>
  7069. /// <returns></returns>
  7070. [HttpPost]
  7071. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7072. public async Task<IActionResult> OverseaVehicleDelAsync(OverseaVehicleDelDto dto)
  7073. {
  7074. return Ok(await _overseaVehicleRep.DelAsync(dto));
  7075. }
  7076. #endregion
  7077. #region 物料成本
  7078. /// <summary>
  7079. /// 物料成本列表查询
  7080. /// </summary>
  7081. /// <param name="dto"></param>
  7082. /// <returns></returns>
  7083. [HttpPost]
  7084. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7085. public async Task<IActionResult> MaterialCostList(QueryMaterialCostDto dto)
  7086. {
  7087. var data = await _materialCostRep.GetListAsync(dto);
  7088. return Ok(JsonView(true, "查询成功", data));
  7089. }
  7090. /// <summary>
  7091. /// 物料成本详情
  7092. /// </summary>
  7093. /// <param name="id"></param>
  7094. /// <returns></returns>
  7095. [HttpGet]
  7096. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7097. public async Task<IActionResult> MaterialCostInfo(int id)
  7098. {
  7099. var data = await _materialCostRep.GetDetailAsync(id);
  7100. return Ok(JsonView(true, "查询成功", data));
  7101. }
  7102. /// <summary>
  7103. /// 物料成本添加/修改
  7104. /// </summary>
  7105. /// <param name="dto"></param>
  7106. /// <returns></returns>
  7107. [HttpPost]
  7108. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7109. public async Task<IActionResult> MaterialCostAddOrEdit(UpMaterialCostDto dto)
  7110. {
  7111. if (dto.Id > 0)
  7112. {
  7113. // 修改
  7114. var entity = _mapper.Map<Res_MaterialCost>(dto);
  7115. var result = await _materialCostRep.UpdateAsync(s => s.Id == dto.Id, it => new Res_MaterialCost()
  7116. {
  7117. TypeId = entity.TypeId,
  7118. Name = entity.Name,
  7119. Price = entity.Price,
  7120. Number = entity.Number,
  7121. Unit = entity.Unit,
  7122. SetDataId = entity.SetDataId,
  7123. CityId = entity.CityId,
  7124. Remark = entity.Remark
  7125. });
  7126. return Ok(JsonView(result, result ? "修改成功" : "修改失败"));
  7127. }
  7128. else
  7129. {
  7130. // 添加
  7131. var entity = _mapper.Map<Res_MaterialCost>(dto);
  7132. entity.CreateTime = DateTime.Now;
  7133. var result = await _materialCostRep.AddAsync(entity);
  7134. return Ok(JsonView(result > 0, result > 0 ? "添加成功" : "添加失败"));
  7135. }
  7136. }
  7137. /// <summary>
  7138. /// 物料成本删除
  7139. /// </summary>
  7140. /// <param name="dto"></param>
  7141. /// <returns></returns>
  7142. [HttpPost]
  7143. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7144. public async Task<IActionResult> MaterialCostDel(DelBaseDto dto)
  7145. {
  7146. var result = await _materialCostRep.SoftDeleteAsync(dto.Id.ToString());
  7147. return Ok(JsonView(result, result ? "删除成功" : "删除失败"));
  7148. }
  7149. #endregion
  7150. #region OP地接资料 - AI
  7151. /// <summary>
  7152. /// OP地接资料AI 基础数据源
  7153. /// </summary>
  7154. /// <returns></returns>
  7155. [HttpGet]
  7156. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7157. public async Task<IActionResult> OpLocalServiceInit()
  7158. {
  7159. var itemNames = await OpLocalServiceAINames();
  7160. var countries = await InvitationAICountryName();
  7161. var orgScales = OrgScale.BuildInitialData().Select(x => x.Name).ToList();
  7162. var orgLevels = new List<string>() {
  7163. "全部",
  7164. "总部",
  7165. "分公司"
  7166. };
  7167. return Ok(JsonView(true, $"查询成功!", new
  7168. {
  7169. itemNames,
  7170. countries,
  7171. orgScales,
  7172. orgLevels
  7173. }));
  7174. }
  7175. /// <summary>
  7176. /// OP地接资料AI 资料列表
  7177. /// </summary>
  7178. /// <returns></returns>
  7179. [HttpGet("{name}")]
  7180. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7181. public async Task<IActionResult> OpLocalServiceAIItemByName(string name)
  7182. {
  7183. if (string.IsNullOrWhiteSpace(name)) return Ok(JsonView(false, "名称无效"));
  7184. // 1. 获取主表记录
  7185. var info = await _sqlSugar.Queryable<Res_OpLocalAI>()
  7186. .FirstAsync(x => x.IsDel == 0 && x.LocalName == name);
  7187. var groupInfo = await _sqlSugar.Queryable<Grp_DelegationInfo>().FirstAsync(x => x.IsDel == 0 && x.TeamName.Equals(name));
  7188. string baseUrl = AppSettingsHelper.Get("OfficeBaseUrl")?.TrimEnd('/');
  7189. // 2. 分支处理:若无主表记录,尝试从团组表同步基本信息
  7190. if (info == null)
  7191. {
  7192. return Ok(JsonView(true, "暂无数据", new
  7193. {
  7194. Id = 0,
  7195. GroupId = groupInfo?.Id ?? 0,
  7196. LocalName = name,
  7197. AiCrawledDetails = new List<LocalAgencyInfo>(),
  7198. Entry = new
  7199. {
  7200. Objective = groupInfo?.VisitPurpose ?? "商务会谈",
  7201. OriginUnit = groupInfo?.ClientUnit ?? "",
  7202. TargetCountry = _delegationInfoRep.GroupSplitCountry(groupInfo?.VisitCountry ?? "")
  7203. }
  7204. }));
  7205. }
  7206. // 3. 数据填充(使用 ??= 语法确保 Null 安全)
  7207. info.EntryInfo ??= new LocalEntryInfo();
  7208. if (string.IsNullOrEmpty(info.EntryInfo.Objective))
  7209. info.EntryInfo.Objective = groupInfo?.VisitPurpose ?? "商务会谈";
  7210. if (string.IsNullOrEmpty(info.EntryInfo.OriginUnit))
  7211. info.EntryInfo.OriginUnit = groupInfo?.ClientUnit ?? "";
  7212. if (info.EntryInfo.TargetCountry?.Any() != true)
  7213. info.EntryInfo.TargetCountry = _delegationInfoRep.GroupSplitCountry(groupInfo?.VisitCountry ?? "");
  7214. // 4. 排序:ThenByDescending 确保多级排序生效
  7215. info.AiCrawledDetails = info.AiCrawledDetails
  7216. .OrderByDescending(x => x.IsChecked)
  7217. .ThenByDescending(x => x.OperatedAt)
  7218. .ToList();
  7219. var details = info?.AiCrawledDetails ?? new List<LocalAgencyInfo>();
  7220. if (details.Any())
  7221. {
  7222. foreach (var detail in details)
  7223. {
  7224. // 附件路径处理
  7225. if (detail.EmailInfo?.AttachmentPaths?.Any() == true)
  7226. {
  7227. detail.EmailInfo.AttachmentPaths = detail.EmailInfo.AttachmentPaths
  7228. .Select(p => p.StartsWith("http") ? p : $"{baseUrl}/{p.TrimStart('/')}")
  7229. .ToList();
  7230. }
  7231. }
  7232. }
  7233. return Ok(JsonView(true, "查询成功!", new
  7234. {
  7235. info.Id,
  7236. info.GroupId,
  7237. info.LocalName,
  7238. AiCrawledDetails = details,
  7239. Entry = info.EntryInfo
  7240. }));
  7241. }
  7242. /// <summary>
  7243. /// OP地接资料AI 设置词条
  7244. /// </summary>
  7245. /// <returns></returns>
  7246. [HttpPost]
  7247. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7248. public async Task<IActionResult> OpLocalServiceAISetPrompt(OpLocalServiceAISetPromptDto dto)
  7249. {
  7250. // 基础校验
  7251. if (string.IsNullOrWhiteSpace(dto.LocalName) || dto.TargetCountry == null || dto.TargetCountry.Count == 0 ||
  7252. dto.ScaleTypes == null || dto.ScaleTypes.Count == 0)
  7253. return Ok(JsonView(false, "请传入有效的地接名称、国家、规模类型!"));
  7254. var localName = dto.LocalName;
  7255. var groupInfo = await _sqlSugar.Queryable<Grp_DelegationInfo>()
  7256. .Where(x => x.IsDel == 0 && x.TeamName.Equals(localName))
  7257. .Select(x => new { x.Id, x.TeamName, x.VisitPurpose })
  7258. .FirstAsync();
  7259. int groupId = groupInfo?.Id ?? 0;
  7260. #region 数据库操作
  7261. // 数据库信息获取方式
  7262. var dataInfo = await _sqlSugar.Queryable<Res_OpLocalAI>().Where(x => x.IsDel == 0 && x.LocalName.Equals(localName)).FirstAsync();
  7263. #region 词条信息
  7264. string operatorName = await _sqlSugar.Queryable<Sys_Users>().Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  7265. var entryInfo = new LocalEntryInfo()
  7266. {
  7267. OriginUnit = dto.OriginUnit,
  7268. TargetCountry = dto.TargetCountry,
  7269. ScaleTypes = dto.ScaleTypes,
  7270. IsBackground = dto.IsBackground,
  7271. OrgLevel = dto.OrgLevel,
  7272. OtherConstraints = dto.OtherConstraints,
  7273. Operator = operatorName,
  7274. OperatedAt = DateTime.Now
  7275. };
  7276. #endregion
  7277. // 3.如果以上两种方式都没有查询到数据,则说明是新数据,需要添加到数据库
  7278. if (dataInfo == null)
  7279. {
  7280. // 3.1 新数据,需要添加到数据库
  7281. dataInfo = new Res_OpLocalAI()
  7282. {
  7283. LocalName = localName,
  7284. GroupId = groupId,
  7285. EntryInfo = entryInfo,
  7286. CreateUserId = dto.CurrUserId
  7287. };
  7288. var insert = await _sqlSugar.Insertable(dataInfo).ExecuteReturnIdentityAsync();
  7289. if (insert < 1)
  7290. {
  7291. return Ok(JsonView(false, $"词条信息新增失败!"));
  7292. }
  7293. dataInfo.Id = insert;
  7294. }
  7295. else
  7296. {
  7297. // 3.2 数据存在 则更新数据(覆盖原有的AI爬取详情,保留原有的其他字段不变)
  7298. dataInfo.EntryInfo = entryInfo;
  7299. var update = await _sqlSugar.Updateable(dataInfo).ExecuteCommandAsync();
  7300. if (update < 1)
  7301. {
  7302. return Ok(JsonView(false, $"词条信息更新失败!"));
  7303. }
  7304. }
  7305. #endregion
  7306. return Ok(JsonView(true, $"设置成功!", new
  7307. {
  7308. dataInfo.Id,
  7309. dataInfo.LocalName,
  7310. dataInfo.GroupId,
  7311. Entry = dataInfo.EntryInfo,
  7312. }));
  7313. }
  7314. /// <summary>
  7315. /// OP地接资料AI 设置复选框选中 批量
  7316. /// </summary>
  7317. /// <param name="dto"></param>
  7318. /// <returns></returns>
  7319. [HttpPost]
  7320. public async Task<IActionResult> OpLocalServiceAISetChecked([FromBody] OpLocalServiceAISetCheckedDto dto)
  7321. {
  7322. // 1. 基础参数校验 (注意:这里移除了对 Guids.Any() 的强校验,允许空集合)
  7323. if (dto.Id < 1 || dto.CurrUserId < 1 || dto.Guids == null)
  7324. return Ok(JsonView(false, "请求参数不完整"));
  7325. // 2. 获取主记录
  7326. var invAiInfo = await _sqlSugar.Queryable<Res_OpLocalAI>()
  7327. .FirstAsync(x => x.IsDel == 0 && x.Id == dto.Id);
  7328. if (invAiInfo?.AiCrawledDetails == null || !invAiInfo.AiCrawledDetails.Any())
  7329. return Ok(JsonView(false, "数据不存在或集合为空"));
  7330. // 3. 准备更新所需数据
  7331. var guidSet = dto.Guids.ToHashSet();
  7332. var now = DateTime.Now;
  7333. // 获取操作人姓名
  7334. string operatorName = await _sqlSugar.Queryable<Sys_Users>()
  7335. .Where(x => x.Id == dto.CurrUserId)
  7336. .Select(x => x.CnName)
  7337. .FirstAsync() ?? "-";
  7338. // 4. 单次遍历完成 重置 + 更新
  7339. // 如果 guidSet 为空,循环会将所有项的 IsChecked 置为 false
  7340. foreach (var item in invAiInfo.AiCrawledDetails)
  7341. {
  7342. if (guidSet.Contains(item.Guid))
  7343. {
  7344. item.IsChecked = true;
  7345. }
  7346. else
  7347. {
  7348. item.IsChecked = false;
  7349. }
  7350. item.OperatedAt = now;
  7351. item.Operator = operatorName;
  7352. }
  7353. // 5. 排序逻辑
  7354. // 即使是全量取消选中,也可以按最后操作时间排序,让最近变动的项在前
  7355. invAiInfo.AiCrawledDetails = invAiInfo.AiCrawledDetails
  7356. .OrderByDescending(x => x.OperatedAt)
  7357. .ToList();
  7358. // 6. 提交数据库
  7359. // 注意:SqlSugar 的 UpdateColumns 会序列化整个对象列表并覆盖数据库字段
  7360. await _sqlSugar.Updateable(invAiInfo)
  7361. .UpdateColumns(x => x.AiCrawledDetails)
  7362. .ExecuteCommandAsync();
  7363. return Ok(JsonView(true, guidSet.Any() ? "设置成功" : "已全部取消选中"));
  7364. }
  7365. /// <summary>
  7366. /// OP地接资料AI 保存
  7367. /// </summary>
  7368. /// <returns></returns>
  7369. [HttpPost]
  7370. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7371. public async Task<IActionResult> OpLocalServiceAISave(OpLocalServiceAISaveDto dto)
  7372. {
  7373. if (dto.Id < 1) return Ok(JsonView(false, "请先设置AI关键字"));
  7374. if (dto.CurrUserId < 1) return Ok(JsonView(false, "请传入用户Id"));
  7375. if (dto.AiCrawledDetail == null) return Ok(JsonView(false, "请选择保存的邀请方详情"));
  7376. var localAiInfo = await _sqlSugar.Queryable<Res_OpLocalAI>()
  7377. .Where(x => x.Id == dto.Id)
  7378. .FirstAsync();
  7379. if (localAiInfo == null) return Ok(JsonView(false, "数据信息为空"));
  7380. var dataList = localAiInfo?.AiCrawledDetails;
  7381. if (dataList == null) return Ok(JsonView(false, "邀请方数据信息为空"));
  7382. var datas = dataList.Where(x => x.Guid != dto.AiCrawledDetail.Guid).ToList();
  7383. var editInfo = dto.AiCrawledDetail;
  7384. // 如果 Guid 为空,说明是新增数据,需要生成新的 Guid
  7385. if (string.IsNullOrEmpty(editInfo.Guid))
  7386. {
  7387. editInfo.Guid = Guid.NewGuid().ToString("N");
  7388. editInfo.Source = 2; // 标记为手动新增数据
  7389. editInfo.OperatedAt = DateTime.Now;
  7390. }
  7391. var opUserName = await _sqlSugar.Queryable<Sys_Users>().Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  7392. editInfo.Operator = opUserName;
  7393. editInfo.OperatedAt = DateTime.Now;
  7394. datas.Add(editInfo);
  7395. localAiInfo.AiCrawledDetails = datas.OrderByDescending(x => x.OperatedAt).ToList();
  7396. var editUpd = await _sqlSugar.Updateable(localAiInfo).UpdateColumns(x => x.AiCrawledDetails).ExecuteCommandAsync();
  7397. if (editUpd < 1)
  7398. {
  7399. return Ok(JsonView(true, "保存失败"));
  7400. }
  7401. return Ok(JsonView(true, "保存成功"));
  7402. }
  7403. /// <summary>
  7404. /// OP地接资料AI 单条删除
  7405. /// </summary>
  7406. /// <returns></returns>
  7407. [HttpPost]
  7408. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7409. public async Task<IActionResult> OpLocalServiceAISingleDel(OpLocalServiceAISingleDelDto dto)
  7410. {
  7411. int id = dto.Id;
  7412. string guid = dto.Guid;
  7413. // 基础校验
  7414. if (id < 1 || string.IsNullOrWhiteSpace(guid))
  7415. return Ok(JsonView(false, "请传入有效的Id和Guid!"));
  7416. var dataInfo = await _sqlSugar.Queryable<Res_OpLocalAI>().Where(x => x.IsDel == 0 && x.Id == id).FirstAsync();
  7417. if (dataInfo == null)
  7418. return Ok(JsonView(false, "当前数据信息不存在!"));
  7419. var newDataInfos = dataInfo.AiCrawledDetails.Where(x => x.Guid != guid).ToList();
  7420. dataInfo.AiCrawledDetails = newDataInfos.OrderByDescending(x => x.OperatedAt).ToList();
  7421. var update = await _sqlSugar.Updateable(dataInfo).ExecuteCommandAsync();
  7422. if (update < 1)
  7423. {
  7424. return Ok(JsonView(false, $"删除失败!"));
  7425. }
  7426. return Ok(JsonView(true, $"删除成功!", new
  7427. {
  7428. dataInfo.Id,
  7429. dataInfo.LocalName,
  7430. dataInfo.GroupId,
  7431. dataInfo.AiCrawledDetails
  7432. }));
  7433. }
  7434. /// <summary>
  7435. /// OP地接资料AI AI查询出的数据添加至商邀资料库
  7436. /// </summary>
  7437. /// <returns></returns>
  7438. [HttpPost]
  7439. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7440. public async Task<IActionResult> OpLocalServiceAIInsertResource([FromBody] OpLocalServiceAIInsertResourceDto dto)
  7441. {
  7442. // 基础校验
  7443. if (dto.Id < 1 || dto.CurrUserId < 1 || dto.Guids == null || !dto.Guids.Any())
  7444. return Ok(JsonView(false, "参数不完整,请选择有效的单位数据"));
  7445. // 获取主记录
  7446. var dataInfo = await _sqlSugar.Queryable<Res_InvitationAI>()
  7447. .FirstAsync(x => x.IsDel == 0 && x.Id == dto.Id);
  7448. if (dataInfo?.AiCrawledDetails == null)
  7449. return Ok(JsonView(false, "当前数据信息不存在!"));
  7450. // 筛选出待转正的 AI 数据 (使用 HashSet 优化匹配)
  7451. var guidSet = dto.Guids.ToHashSet();
  7452. var targetAiDetails = dataInfo.AiCrawledDetails
  7453. .Where(x => x.Source != 0 && guidSet.Contains(x.Guid))
  7454. .ToList();
  7455. if (!targetAiDetails.Any())
  7456. return Ok(JsonView(false, "未找到符合条件的 AI 获取信息,无法重复添加或来源错误!"));
  7457. // 准备入库数据(批量构建)
  7458. var now = DateTime.Now;
  7459. var opUserName = await _sqlSugar.Queryable<Sys_Users>().Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  7460. var rawLocalDatas = targetAiDetails.Select(item => {
  7461. // 修改内存中的状态为本地数据
  7462. item.Source = 0;
  7463. item.OperatedAt = now;
  7464. item.Operator = opUserName;
  7465. return new Res_InvitationOfficialActivityData()
  7466. {
  7467. // 批量加密操作
  7468. Country = AesEncryptionHelper.Encrypt(item.Region),
  7469. UnitName = AesEncryptionHelper.Encrypt(item.NameCn),
  7470. Address = AesEncryptionHelper.Encrypt(item.Address),
  7471. Field = AesEncryptionHelper.Encrypt(item.Scope),
  7472. Contact = AesEncryptionHelper.Encrypt(item.Contact),
  7473. Tel = AesEncryptionHelper.Encrypt(item.Phone),
  7474. Email = AesEncryptionHelper.Encrypt(item.Email),
  7475. UnitWeb = AesEncryptionHelper.Encrypt(item.SiteUrl),
  7476. LastUpdateUserId = dto.CurrUserId,
  7477. LastUpdateTime = now,
  7478. CreateUserId = dto.CurrUserId,
  7479. CreateTime = now
  7480. };
  7481. }).ToList();
  7482. // 开启事务执行双表更新
  7483. var result = await _sqlSugar.UseTranAsync(async () =>
  7484. {
  7485. // 插入本地资料库
  7486. await _sqlSugar.Insertable(rawLocalDatas).ExecuteCommandAsync();
  7487. // 更新 AI 主表的状态(全量覆盖 JSON 列)
  7488. await _sqlSugar.Updateable(dataInfo)
  7489. .UpdateColumns(x => x.AiCrawledDetails)
  7490. .ExecuteCommandAsync();
  7491. });
  7492. return Ok(result.IsSuccess
  7493. ? JsonView(true, "成功添加至资料库并更新来源状态")
  7494. : JsonView(false, $"操作失败:{result.ErrorMessage}"));
  7495. }
  7496. /// <summary>
  7497. /// OP地接资料AI 混元AI查询资料(SSE流式推送)
  7498. /// </summary>
  7499. [HttpPost]
  7500. public async Task OpLocalServiceAISearchStream([FromBody] OpLocalServiceAISearchDto dto)
  7501. {
  7502. HttpContext.InitializeSse();
  7503. try
  7504. {
  7505. await HttpContext.SendSseStepAsync(5, "正在加载配置...");
  7506. #region 1. 异步并行化验证 (Performance Boost)
  7507. var localAiInfo = await _sqlSugar.Queryable<Res_OpLocalAI>().Where(x => x.IsDel == 0 && x.Id == dto.Id).FirstAsync();
  7508. var operatorName = await _sqlSugar.Queryable<Sys_Users>()
  7509. .Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId)
  7510. .Select(x => x.CnName).FirstAsync();
  7511. if (localAiInfo?.EntryInfo == null)
  7512. {
  7513. await HttpContext.SendSseStepAsync(-1, "未找到有效的关键字配置。");
  7514. return;
  7515. }
  7516. var entryInfo = localAiInfo.EntryInfo;
  7517. var targetCountrySet = new HashSet<string>(entryInfo.TargetCountry);
  7518. #endregion
  7519. await HttpContext.SendSseStepAsync(20, "正在解析本地典籍 (并行解密)...");
  7520. #region 2. 内存计算优化
  7521. //// 仅查询必要字段,减少 DataReader 压力
  7522. //var rawLocalDatas = await _sqlSugar.Queryable<Res_InvitationOfficialActivityData>()
  7523. // .Where(x => x.IsDel == 0 && !string.IsNullOrEmpty(x.UnitName))
  7524. // .Select(x => new { x.Country, x.UnitName, x.Address, x.Field, x.Contact, x.Tel, x.Email })
  7525. // .ToListAsync();
  7526. //// PLINQ 核心炼金:利用多核并行解密
  7527. //var allDecrypted = rawLocalDatas.AsParallel().Select(item => new InvitationAIInfo
  7528. //{
  7529. // Guid = Guid.NewGuid().ToString("N"),
  7530. // Source = 0, // 标识来源:本地
  7531. // Region = AesEncryptionHelper.Decrypt(item.Country),
  7532. // NameCn = AesEncryptionHelper.Decrypt(item.UnitName),
  7533. // Address = AesEncryptionHelper.Decrypt(item.Address),
  7534. // Scope = AesEncryptionHelper.Decrypt(item.Field),
  7535. // Contact = AesEncryptionHelper.Decrypt(item.Contact),
  7536. // Phone = AesEncryptionHelper.Decrypt(item.Tel),
  7537. // Email = AesEncryptionHelper.Decrypt(item.Email),
  7538. // OperatedAt = DateTime.Now,
  7539. // Operator = operatorName,
  7540. //}).ToList();
  7541. //// 筛选符合国家的本地数据
  7542. //var matchedCountries = allDecrypted.Where(x => targetCountrySet.Contains(x.Region)).ToList();
  7543. var matchedCountries = new List<LocalAgencyInfo>();
  7544. #endregion
  7545. #region 3. 混元 AI 协同炼金
  7546. // --- 差额补全 (Gap Filling) ---
  7547. var localDatas = new List<LocalAgencyInfo>();
  7548. var aiTasks = new List<CountryAIPormptInfo>();
  7549. if (targetCountrySet.Any())
  7550. {
  7551. foreach (var item in targetCountrySet)
  7552. {
  7553. aiTasks.Add(new CountryAIPormptInfo() {
  7554. Count = entryInfo.NeedCount,
  7555. Country = item
  7556. });
  7557. }
  7558. }
  7559. var hunyuanAIInvDatas = new List<LocalAgencyInfo>();
  7560. if (aiTasks.Any())
  7561. {
  7562. await HttpContext.SendSseStepAsync(60, $"AI 正在跨境检索 {string.Join(", ", aiTasks.Select(x => $"{x.Country}({x.Count}条)"))} 单位资料...");
  7563. string searchQuestion = OpLocalBuildHunyuanPrompt(aiTasks, entryInfo);
  7564. _logger.LogInformation(@"地接名称:{InvName}; 混元AI查询提示词:{searchQuestion}", localAiInfo.LocalName, searchQuestion);
  7565. string searchRaw = await _hunyuanService.ChatCompletionsHunyuan_t1_latestAsync(searchQuestion);
  7566. var aiParsed = CleanAndParseJson<List<LocalAgencyInfo>>(searchRaw);
  7567. if (aiParsed != null)
  7568. {
  7569. hunyuanAIInvDatas = aiParsed.Select(x => {
  7570. x.Guid = Guid.NewGuid().ToString("N");
  7571. x.Source = 1;
  7572. x.Operator = operatorName;
  7573. x.OperatedAt = DateTime.Now;
  7574. return x;
  7575. }).ToList();
  7576. }
  7577. }
  7578. #endregion
  7579. await HttpContext.SendSseStepAsync(90, "正在同步成果至持久化仓库...");
  7580. #region 4. 数据融合与更新
  7581. var finalResult = localDatas.Concat(hunyuanAIInvDatas).ToList();
  7582. // 仅更新指定列,性能更优
  7583. localAiInfo.AiCrawledDetails = finalResult.OrderByDescending(x => x.OperatedAt).ToList();
  7584. bool isOk = await _sqlSugar.Updateable(localAiInfo)
  7585. .UpdateColumns(x => x.AiCrawledDetails)
  7586. .ExecuteCommandHasChangeAsync();
  7587. if (!isOk) await HttpContext.SendSseStepAsync(-1, $"数据持久化失败。");
  7588. #endregion
  7589. // 5. 终焉推送
  7590. await HttpContext.SendSseStepAsync(100, "操作成功!资料已全部就绪。", new
  7591. {
  7592. localAiInfo.Id,
  7593. AiCrawledDetails = finalResult.OrderByDescending(x => x.Source).ThenBy(x => x.Region).ToList()
  7594. });
  7595. }
  7596. catch (Exception ex)
  7597. {
  7598. _logger.LogError(ex, "SSE 管道熔断");
  7599. await HttpContext.SendSseStepAsync(-1, $"SSE 错误:{ex.InnerException.Message}({ex.Message})");
  7600. }
  7601. finally
  7602. {
  7603. await HttpContext.FinalizeSseAsync();
  7604. }
  7605. }
  7606. /// <summary>
  7607. /// OP地接资料AI 混元AI续写(SSE流式推送)
  7608. /// </summary>
  7609. /// <param name="dto"></param>
  7610. /// <returns></returns>
  7611. [HttpPost]
  7612. public async Task OpLocalServiceAICompleteTextStream([FromBody] OpLocalServiceAICompleteTextStreamDto dto)
  7613. {
  7614. HttpContext.InitializeSse();
  7615. try
  7616. {
  7617. // 初始化检查 (Progress: 5%)
  7618. var invAiInfo = await _sqlSugar.Queryable<Res_OpLocalAI>().Where(x => x.Id == dto.Id).FirstAsync();
  7619. if (invAiInfo?.EntryInfo == null)
  7620. {
  7621. await HttpContext.SendSseStepAsync(-1, "请先设置关键字信息!");
  7622. return;
  7623. }
  7624. await HttpContext.SendSseStepAsync(10, "任务初始化完成,正在计算补齐缺口...");
  7625. // 任务拆解逻辑
  7626. var entryInfo = invAiInfo.EntryInfo;
  7627. var aiTasks = new List<CountryAIPormptInfo>();
  7628. int targetPerCountry = entryInfo.NeedCount;
  7629. foreach (var countryName in entryInfo.TargetCountry)
  7630. {
  7631. var countryDataCount = invAiInfo.AiCrawledDetails.Count(x => x.Region == countryName);
  7632. int aiNeedCount = targetPerCountry - countryDataCount;
  7633. if (aiNeedCount > 0) aiTasks.Add(new() { Country = countryName, Count = aiNeedCount });
  7634. }
  7635. if (!aiTasks.Any())
  7636. {
  7637. await HttpContext.SendSseStepAsync(100, "数据已满额,无需续写", invAiInfo.AiCrawledDetails);
  7638. return;
  7639. }
  7640. // 准备 AI Prompt (Progress: 20%)
  7641. var existingNames = new HashSet<string>(invAiInfo.AiCrawledDetails.Where(x => x.Source == 1).Select(x => x.NameCn));
  7642. string promptOther = $"请基于以下已存在的名称列表进行推荐,避免重复:{string.Join(", ", existingNames)}。{entryInfo.OtherConstraints}";
  7643. entryInfo.OtherConstraints = promptOther; // 将去重提示注入 entryInfo,确保 Prompt 构建时包含该信息
  7644. // 构建 Question
  7645. string question = OpLocalBuildHunyuanPrompt(aiTasks, entryInfo);
  7646. await HttpContext.SendSseStepAsync(30, "AI 正在深度检索跨境OP地接数据,请稍候...");
  7647. // 调用 AI (Progress: 30% - 80%)
  7648. string aiRawResponse = await _hunyuanService.ChatCompletionsHunyuan_t1_latestAsync(question);
  7649. await HttpContext.SendSseStepAsync(85, "数据已捕获,正在进行格式校验与去重...");
  7650. // 5. 解析与清洗数据
  7651. var hunyuanAIInvDatas = new List<LocalAgencyInfo>();
  7652. if (!string.IsNullOrWhiteSpace(aiRawResponse))
  7653. {
  7654. string cleanJson = aiRawResponse.Trim();
  7655. if (cleanJson.Contains("```json"))
  7656. {
  7657. cleanJson = Regex.Match(cleanJson, @"```json([\s\S]*?)```").Groups[1].Value.Trim();
  7658. }
  7659. else if (cleanJson.Contains("```"))
  7660. {
  7661. cleanJson = Regex.Match(cleanJson, @"```([\s\S]*?)```").Groups[1].Value.Trim();
  7662. }
  7663. hunyuanAIInvDatas = JsonConvert.DeserializeObject<List<LocalAgencyInfo>>(cleanJson) ?? new List<LocalAgencyInfo>();
  7664. }
  7665. string operatorName = await _sqlSugar.Queryable<Sys_Users>().Where(x => x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  7666. foreach (var x in hunyuanAIInvDatas)
  7667. {
  7668. x.Guid = Guid.NewGuid().ToString("N");
  7669. x.Source = 1;
  7670. x.Operator = operatorName;
  7671. x.OperatedAt = DateTime.Now;
  7672. }
  7673. // 6. 数据库操作 (Progress: 95%)
  7674. invAiInfo.AiCrawledDetails.AddRange(hunyuanAIInvDatas);
  7675. invAiInfo.AiCrawledDetails = invAiInfo.AiCrawledDetails.OrderByDescending(x => x.OperatedAt).ToList();
  7676. var update = await _sqlSugar.Updateable(invAiInfo).UpdateColumns(x => x.AiCrawledDetails).ExecuteCommandAsync();
  7677. if (update > 0)
  7678. {
  7679. await HttpContext.SendSseStepAsync(100, $"AI 续写成功!新增 {hunyuanAIInvDatas.Count} 条数据", invAiInfo.AiCrawledDetails);
  7680. }
  7681. else
  7682. {
  7683. await HttpContext.SendSseStepAsync(-1, "数据库更新失败");
  7684. }
  7685. }
  7686. catch (Exception ex)
  7687. {
  7688. _logger.LogError(ex, "SSE 续写异常");
  7689. await HttpContext.SendSseStepAsync(-1, $"炼金炸炉:{ex.Message}");
  7690. }
  7691. finally
  7692. {
  7693. await HttpContext.FinalizeSseAsync();
  7694. }
  7695. }
  7696. /// <summary>
  7697. /// OP地接资料AI 生成报价邮件(SSE流式推送)
  7698. /// </summary>
  7699. /// <returns></returns>
  7700. [HttpPost]
  7701. public async Task OpLocalServiceAIGenerateEmailStream([FromBody] OpLocalServiceAIGenerateEmailStreamDto dto)
  7702. {
  7703. HttpContext.InitializeSse();
  7704. try
  7705. {
  7706. // 1. 基础校验与资源加载 (Progress: 5%)
  7707. if (dto.Id < 1 || dto.Guids == null || !dto.Guids.Any())
  7708. {
  7709. await HttpContext.SendSseStepAsync(-1, "请求参数不完整,请选择单位");
  7710. return;
  7711. }
  7712. var localAiInfo = await _sqlSugar.Queryable<Res_OpLocalAI>().Where(x => x.Id == dto.Id).FirstAsync();
  7713. if (localAiInfo?.AiCrawledDetails == null)
  7714. {
  7715. await HttpContext.SendSseStepAsync(-1, "基础信息不存在");
  7716. return;
  7717. }
  7718. await HttpContext.SendSseStepAsync(10, "AI 正在分析考察团组背景与访问意图...");
  7719. // 2. 准备 AI 上下文数据 (Progress: 15%)
  7720. var clientInfoSources = localAiInfo.AiCrawledDetails.Where(x => dto.Guids.Contains(x.Guid)).ToList();
  7721. var groupInfo = await _sqlSugar.Queryable<Grp_DelegationInfo>()
  7722. .Where(x => x.IsDel == 0 && x.Id == localAiInfo.GroupId)
  7723. .Select(x => new { x.TeamName, x.VisitPNumber, x.VisitDate, x.VisitStartDate, x.VisitEndDate, x.VisitCountry })
  7724. .FirstAsync();
  7725. string dateRange = $"{groupInfo.VisitStartDate:yyyy-MM-dd} ~ {groupInfo.VisitEndDate:yyyy-MM-dd}";
  7726. string operatorName = await _sqlSugar.Queryable<Sys_Users>()
  7727. .Where(x => x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  7728. // 3. 构建 Prompt 并调用 AI (Progress: 25%)
  7729. await HttpContext.SendSseStepAsync(30, $"AI 正在为 {clientInfoSources.Count} 家单位撰写定制化正式邮件...");
  7730. // 构建生成邮件的 Prompt
  7731. string pormpt = BuildBatchCreateEmailPrompt(
  7732. new AICreateEmailTask() {
  7733. PersonCount = groupInfo.VisitPNumber,
  7734. DateRange = dateRange,
  7735. Agencies = clientInfoSources,
  7736. EntryInfo = localAiInfo.EntryInfo
  7737. }
  7738. );
  7739. // 调用 AI (此处为阻塞式等待 AI 结果,若混元支持 Stream 可进一步拆解)
  7740. string aiResponse = await _hunyuanService.ChatCompletionsHunyuan_t1_latestAsync(pormpt);
  7741. await HttpContext.SendSseStepAsync(80, "邮件初稿已生成,正在进行 HTML 格式校验与转义处理...");
  7742. // 4. 解析结果 (Progress: 85%)
  7743. var hunyuanAIEmailDatas = new List<AICreateEmailInfo>();
  7744. if (!string.IsNullOrWhiteSpace(aiResponse))
  7745. {
  7746. // 预处理:过滤 AI 可能返回的 Markdown 标记
  7747. string cleanJson = aiResponse.Trim();
  7748. if (cleanJson.StartsWith("```json"))
  7749. cleanJson = cleanJson.Substring(7, cleanJson.Length - 10).Trim();
  7750. else if (cleanJson.StartsWith("```"))
  7751. cleanJson = cleanJson.Substring(3, cleanJson.Length - 6).Trim();
  7752. try
  7753. {
  7754. // 解析并注入 Source 标识
  7755. hunyuanAIEmailDatas = JsonConvert.DeserializeObject<List<AICreateEmailInfo>>(cleanJson);
  7756. }
  7757. catch (JsonException ex)
  7758. {
  7759. // 记录日志并考虑 fallback 策略
  7760. _logger.LogError(ex, "Hunyuan AI 响应解析失败。原始数据:{Response}", aiResponse);
  7761. }
  7762. }
  7763. if (hunyuanAIEmailDatas == null || !hunyuanAIEmailDatas.Any())
  7764. {
  7765. await HttpContext.SendSseStepAsync(-1, "AI 格式解析失败,请尝试重新生成");
  7766. return;
  7767. }
  7768. // 5. 更新数据模型 (Progress: 90%)
  7769. foreach (var client in clientInfoSources)
  7770. {
  7771. var aiEmail = hunyuanAIEmailDatas.FirstOrDefault(x => x.Guid == client.Guid);
  7772. if (aiEmail != null)
  7773. {
  7774. client.EmailInfo.Status = 2; // 已生成
  7775. client.EmailInfo.EmailTitle = aiEmail.Subject;
  7776. client.EmailInfo.EmailContent = aiEmail.Content;
  7777. client.EmailInfo.Operator = operatorName;
  7778. client.EmailInfo.OperatedAt = DateTime.Now;
  7779. }
  7780. }
  7781. localAiInfo.AiCrawledDetails = clientInfoSources.OrderByDescending(x => x.IsChecked).ThenByDescending(x => x.OperatedAt).ToList();
  7782. // 6. 数据库持久化 (Progress: 95%)
  7783. // 采用增量更新策略,避免直接 Where 过滤掉其他未选中的数据
  7784. var update = await _sqlSugar.Updateable(localAiInfo).UpdateColumns(x => x.AiCrawledDetails).ExecuteCommandAsync();
  7785. if (update > 0)
  7786. {
  7787. await HttpContext.SendSseStepAsync(100, "邮件全部生成完毕并已存入团组资料库", localAiInfo.AiCrawledDetails);
  7788. }
  7789. else
  7790. {
  7791. await HttpContext.SendSseStepAsync(-1, "数据库写入失败,请联系管理员");
  7792. }
  7793. }
  7794. catch (Exception ex)
  7795. {
  7796. _logger.LogError(ex, "邮件生成异常");
  7797. await HttpContext.SendSseStepAsync(-1, $"生成失败:{ex.Message}");
  7798. }
  7799. finally
  7800. {
  7801. await HttpContext.FinalizeSseAsync();
  7802. }
  7803. }
  7804. /// <summary>
  7805. /// OP地接资料AI 邮件保存
  7806. /// </summary>
  7807. /// <returns></returns>
  7808. [HttpPost]
  7809. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7810. public async Task<IActionResult> OpLocalServiceAIEmailSave(OpLocalServiceAIEmailSaveDto dto)
  7811. {
  7812. if (dto.Id < 1) return Ok(JsonView(false, "请选择保存的团组"));
  7813. if (dto.CurrUserId < 1) return Ok(JsonView(false, "请传入用户Id"));
  7814. if (string.IsNullOrEmpty(dto.Guid)) return Ok(JsonView(false, "请传入Guid"));
  7815. if (string.IsNullOrEmpty(dto.EmailTitle)) return Ok(JsonView(false, "请传入邮件标题"));
  7816. if (string.IsNullOrEmpty(dto.EmailContent)) return Ok(JsonView(false, "请传入邮件内容"));
  7817. var opLocalAiInfo = await _sqlSugar.Queryable<Res_OpLocalAI>()
  7818. .Where(x => x.Id == dto.Id)
  7819. .FirstAsync();
  7820. if (opLocalAiInfo == null) return Ok(JsonView(false, "地接资料方信息为空"));
  7821. var dataList = opLocalAiInfo?.AiCrawledDetails;
  7822. if (dataList == null) return Ok(JsonView(false, "地接资料方信息为空"));
  7823. var editInfo = dataList.Where(x => x.Guid == dto.Guid).FirstOrDefault();
  7824. if (editInfo == null) return Ok(JsonView(false, "地接资料方信息为空"));
  7825. var opUserName = await _sqlSugar.Queryable<Sys_Users>().Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId).Select(x => x.CnName).FirstAsync() ?? "-";
  7826. editInfo.EmailInfo.EmailTitle = dto.EmailTitle;
  7827. editInfo.EmailInfo.EmailContent = dto.EmailContent;
  7828. editInfo.EmailInfo.Operator = opUserName;
  7829. editInfo.EmailInfo.OperatedAt = DateTime.Now;
  7830. var datas = dataList.Where(x => x.Guid != dto.Guid).ToList();
  7831. datas.Add(editInfo);
  7832. // 排序
  7833. opLocalAiInfo.AiCrawledDetails = datas.OrderByDescending(x => x.OperatedAt).ToList();
  7834. var editUpd = await _sqlSugar.Updateable(opLocalAiInfo).UpdateColumns(x => x.AiCrawledDetails).ExecuteCommandAsync();
  7835. if (editUpd < 1)
  7836. {
  7837. return Ok(JsonView(true, "邮件保存失败"));
  7838. }
  7839. return Ok(JsonView(true, "邮件保存成功"));
  7840. }
  7841. /// <summary>
  7842. /// OP地接资料AI 邮箱附件上传
  7843. /// </summary>
  7844. [HttpPost]
  7845. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7846. public async Task<IActionResult> OpLocalServiceAIFileSave([FromForm] OpLocalServiceAIFileSaveDto dto)
  7847. {
  7848. // 1. 炼金前置:严格参数校验
  7849. if (dto.Id < 1) return Ok(JsonView(false, "请选择保存的团组"));
  7850. if (dto.CurrUserId < 1) return Ok(JsonView(false, "请传入用户Id"));
  7851. if (string.IsNullOrEmpty(dto.Guid)) return Ok(JsonView(false, "请传入Guid"));
  7852. if (dto.Attachments == null || !dto.Attachments.Any()) return Ok(JsonView(false, "请传入附件"));
  7853. // 2. 数据获取:利用 SqlSugar 异步查询
  7854. var opAiInfo = await _sqlSugar.Queryable<Res_OpLocalAI>()
  7855. .InSingleAsync(dto.Id);
  7856. if (opAiInfo?.AiCrawledDetails == null) return Ok(JsonView(false, "地接方信息不存在"));
  7857. // 3. 业务定位:定位到具体需要编辑的 Guid 记录
  7858. var editInfo = opAiInfo.AiCrawledDetails.FirstOrDefault(x => x.Guid == dto.Guid);
  7859. if (editInfo == null) return Ok(JsonView(false, "未找到匹配的 Guid 记录"));
  7860. // 4. 追踪信息更新
  7861. var opUserName = await _sqlSugar.Queryable<Sys_Users>()
  7862. .Where(x => x.Id == dto.CurrUserId && x.IsDel == 0)
  7863. .Select(x => x.CnName)
  7864. .FirstAsync() ?? "-";
  7865. // 更新操作人与时间 (利用引用类型特性)
  7866. editInfo.Operator = opUserName;
  7867. editInfo.OperatedAt = DateTime.Now;
  7868. // 初始化 EmailInfo 确保不为 null
  7869. editInfo.EmailInfo ??= new LocalEmailInfo();
  7870. editInfo.EmailInfo.Operator = opUserName;
  7871. editInfo.EmailInfo.OperatedAt = DateTime.Now;
  7872. editInfo.EmailInfo.AttachmentPaths ??= new List<string>();
  7873. // 5. 构建绝对路径与相对路径
  7874. string dirName = $"{editInfo.NameEn?.Trim() ?? "Default"}_{dto.Guid}";
  7875. string baseDir = AppSettingsHelper.Get("OpLocalServiceAIAssistBasePath");
  7876. string ftpBase = AppSettingsHelper.Get("OpLocalServiceAIAssistFtpPath");
  7877. string absolutePath = Path.Combine(baseDir, dirName);
  7878. string relativePathPrefix = $"{ftpBase}{dirName}/";
  7879. // 核心修复:确保父级目录递归创建,防止 DirectoryNotFoundException
  7880. if (!Directory.Exists(absolutePath))
  7881. {
  7882. Directory.CreateDirectory(absolutePath);
  7883. }
  7884. var newSavedRelativePaths = new List<string>();
  7885. // 6. 持续文件存储与实时验证
  7886. foreach (var file in dto.Attachments)
  7887. {
  7888. // 文件名安全过滤
  7889. string safeName = string.Join("_", file.FileName.Split(Path.GetInvalidFileNameChars()));
  7890. string fullPath = Path.Combine(absolutePath, safeName);
  7891. // 重名冲突处理:文件名(n).ext
  7892. if (System.IO.File.Exists(fullPath))
  7893. {
  7894. string fileNameOnly = Path.GetFileNameWithoutExtension(safeName);
  7895. string extension = Path.GetExtension(safeName);
  7896. int count = 1;
  7897. while (System.IO.File.Exists(fullPath))
  7898. {
  7899. safeName = $"{fileNameOnly}({count++}){extension}";
  7900. fullPath = Path.Combine(absolutePath, safeName);
  7901. }
  7902. }
  7903. try
  7904. {
  7905. // 写入流:使用作用域隔离,确保退出 using 时立即释放句柄
  7906. using (var stream = new FileStream(fullPath, FileMode.Create, FileAccess.Write, FileShare.None))
  7907. {
  7908. await file.CopyToAsync(stream);
  7909. await stream.FlushAsync(); // 强制落盘
  7910. }
  7911. // 实时验证:物理验证 + 逻辑校验
  7912. var fileInfo = new FileInfo(fullPath);
  7913. if (fileInfo.Exists && fileInfo.Length == file.Length)
  7914. {
  7915. newSavedRelativePaths.Add($"{relativePathPrefix}{safeName}");
  7916. }
  7917. }
  7918. catch (Exception ex)
  7919. {
  7920. // 发生 IO 异常时清理残余文件并抛出,触发外层处理
  7921. if (System.IO.File.Exists(fullPath)) System.IO.File.Delete(fullPath);
  7922. // 此处记录日志:
  7923. _logger.LogError(ex, "File Save Error");
  7924. continue; // 跳过失败的文件,继续下一个
  7925. }
  7926. }
  7927. if (!newSavedRelativePaths.Any()) return Ok(JsonView(false, "所有文件上传均失败"));
  7928. // 7. 数据合并与持久化:
  7929. // 将新路径与旧路径合并,并去重
  7930. var updatedPaths = editInfo.EmailInfo.AttachmentPaths;
  7931. updatedPaths.AddRange(newSavedRelativePaths);
  7932. editInfo.EmailInfo.AttachmentPaths = updatedPaths.Distinct().ToList();
  7933. // 重新排序整体列表
  7934. opAiInfo.AiCrawledDetails = opAiInfo.AiCrawledDetails
  7935. .OrderByDescending(x => x.OperatedAt)
  7936. .ToList();
  7937. // SqlSugar 高效部分列更新
  7938. var isSuccess = await _sqlSugar.Updateable(opAiInfo)
  7939. .UpdateColumns(x => x.AiCrawledDetails)
  7940. .ExecuteCommandAsync() > 0;
  7941. if (!isSuccess) return Ok(JsonView(false, "数据库更新失败"));
  7942. // 8. 返回前端数据 (支持 lowerCamelCase)
  7943. var officeBaseUrl = AppSettingsHelper.Get("OfficeBaseUrl");
  7944. var finalResultUrls = editInfo.EmailInfo.AttachmentPaths
  7945. .Select(path => $"{officeBaseUrl}{path}")
  7946. .ToList();
  7947. return Ok(JsonView(true, "邮件附件保存成功", finalResultUrls));
  7948. }
  7949. /// <summary>
  7950. /// OP地接资料AI 邮箱附件删除
  7951. /// </summary>
  7952. [HttpPost]
  7953. [ProducesResponseType(typeof(JsonView), StatusCodes.Status200OK)]
  7954. public async Task<IActionResult> OpLocalServiceAIFileDel([FromBody] OpLocalServiceAIFileDelDto dto)
  7955. {
  7956. // 1. 基础校验
  7957. if (dto.Id < 1) return Ok(JsonView(false, "请选择团组"));
  7958. if (string.IsNullOrEmpty(dto.Guid)) return Ok(JsonView(false, "请传入Guid"));
  7959. if (dto.AttachmentNames == null || !dto.AttachmentNames.Any()) return Ok(JsonView(false, "请传入待删除附件名称"));
  7960. // 2. 数据获取
  7961. var invAiInfo = await _sqlSugar.Queryable<Res_OpLocalAI>().InSingleAsync(dto.Id);
  7962. if (invAiInfo?.AiCrawledDetails == null) return Ok(JsonView(false, "业务数据不存在"));
  7963. var editInfo = invAiInfo.AiCrawledDetails.FirstOrDefault(x => x.Guid == dto.Guid);
  7964. if (editInfo?.EmailInfo?.AttachmentPaths == null) return Ok(JsonView(false, "未找到对应的附件记录"));
  7965. // 3. 操作人追踪
  7966. var opUserName = await _sqlSugar.Queryable<Sys_Users>()
  7967. .Where(x => x.Id == dto.CurrUserId && x.IsDel == 0)
  7968. .Select(x => x.CnName).FirstAsync() ?? "System";
  7969. // 更新追踪状态
  7970. editInfo.Operator = editInfo.EmailInfo.Operator = opUserName;
  7971. editInfo.OperatedAt = editInfo.EmailInfo.OperatedAt = DateTime.Now;
  7972. // 4. 路径
  7973. string dirName = $"{editInfo.NameEn?.Trim() ?? "Default"}_{dto.Guid}";
  7974. string absoluteDir = Path.Combine(AppSettingsHelper.Get("OpLocalServiceAIAssistBasePath"), dirName);
  7975. // 获取当前数据库中的相对路径列表
  7976. var currentPaths = editInfo.EmailInfo.AttachmentPaths;
  7977. bool isChanged = false;
  7978. // 5. 核心删除逻辑
  7979. foreach (var fileName in dto.AttachmentNames)
  7980. {
  7981. // 查找数据库中是否存在包含该文件名的路径 (忽略大小写比较)
  7982. var targetRelativePath = currentPaths.FirstOrDefault(p => p.EndsWith("/" + fileName) || p.Equals(fileName));
  7983. if (targetRelativePath != null)
  7984. {
  7985. // A. 从数据库记录中移除
  7986. currentPaths.Remove(targetRelativePath);
  7987. isChanged = true;
  7988. // B. 物理文件删除逻辑
  7989. // 注意:fullPath 必须是 [基础路径] + [目录名] + [纯文件名]
  7990. string fullPath = Path.Combine(absoluteDir, fileName);
  7991. try
  7992. {
  7993. if (System.IO.File.Exists(fullPath))
  7994. {
  7995. System.IO.File.Delete(fullPath);
  7996. }
  7997. }
  7998. catch (Exception ex)
  7999. {
  8000. // 记录 IO 异常但不中断流程
  8001. _logger.LogWarning($"物理文件删除失败: {fullPath}, {ex.Message}");
  8002. }
  8003. }
  8004. }
  8005. if (!isChanged) return Ok(JsonView(false, "未找到匹配的可删除附件"));
  8006. // 6. 数据同步与持久化
  8007. editInfo.EmailInfo.AttachmentPaths = currentPaths;
  8008. // 排序 (利用引用类型,无需手动 Add/Remove)
  8009. invAiInfo.AiCrawledDetails = invAiInfo.AiCrawledDetails.OrderByDescending(x => x.OperatedAt).ToList();
  8010. var isUpd = await _sqlSugar.Updateable(invAiInfo)
  8011. .UpdateColumns(x => x.AiCrawledDetails)
  8012. .ExecuteCommandAsync() > 0;
  8013. if (!isUpd) return Ok(JsonView(false, "数据库记录更新失败"));
  8014. // 7. 返回剩余附件的完整访问地址
  8015. var officeBaseUrl = AppSettingsHelper.Get("OfficeBaseUrl");
  8016. var remainingUrls = editInfo.EmailInfo.AttachmentPaths
  8017. .Select(path => $"{officeBaseUrl}{path}")
  8018. .ToList();
  8019. return Ok(JsonView(true, "附件删除成功", remainingUrls));
  8020. }
  8021. /// <summary>
  8022. /// OP地接资料AI 发送邮件(SSE 流式推送)
  8023. /// </summary>
  8024. [HttpPost]
  8025. public async Task OpLocalServiceAISeedEmailStream([FromBody] OpLocalServiceAISeedEmailStreamDto dto)
  8026. {
  8027. // 1. 初始化 SSE
  8028. HttpContext.InitializeSse();
  8029. try
  8030. {
  8031. await HttpContext.SendSseStepAsync(5, "正在准备发送邮件队列...");
  8032. #region 1. 参数与权限前置校验
  8033. if (dto.Id < 1 || dto.CurrUserId < 1 || dto.Guids == null || !dto.Guids.Any())
  8034. {
  8035. await HttpContext.SendSseStepAsync(-1, "参数验证失败,请检查选择的单位及用户状态。");
  8036. return;
  8037. }
  8038. // hotmail 配置信息验证
  8039. var hotmailConfig = await _hotmailService.GetUserMailConfig(dto.CurrUserId);
  8040. (bool verify, string msg) = _hotmailService.ConfigVerify(hotmailConfig);
  8041. if (!verify)
  8042. {
  8043. await HttpContext.SendSseStepAsync(-1, msg);
  8044. return;
  8045. }
  8046. // 获取地接信息和用户信息
  8047. var opAiInfo = await _sqlSugar.Queryable<Res_OpLocalAI>().FirstAsync(x => x.IsDel == 0 && x.Id == dto.Id);
  8048. var userInfo = await _sqlSugar.Queryable<Sys_Users>()
  8049. .Where(x => x.IsDel == 0 && x.Id == dto.CurrUserId)
  8050. .Select(x => new { x.Email, x.CnName })
  8051. .FirstAsync();
  8052. if (opAiInfo?.AiCrawledDetails == null)
  8053. {
  8054. await HttpContext.SendSseStepAsync(-1, "未找到有效的地接方数据。");
  8055. return;
  8056. }
  8057. // 提取待发送的目标集合
  8058. var guidSet = new HashSet<string>(dto.Guids);
  8059. var seedInvInfos = opAiInfo.AiCrawledDetails.Where(x => guidSet.Contains(x.Guid)).ToList();
  8060. if (!seedInvInfos.Any())
  8061. {
  8062. await HttpContext.SendSseStepAsync(-1, "所选单位信息在原始库中不存在。");
  8063. return;
  8064. }
  8065. #endregion
  8066. await HttpContext.SendSseStepAsync(15, $"准备就绪,共计 {seedInvInfos.Count} 封邮件待发送...");
  8067. #region 2. 批量发送逻辑 (流式反馈)
  8068. int total = seedInvInfos.Count;
  8069. int current = 0;
  8070. var successCount = 0;
  8071. var failCount = 0;
  8072. foreach (var item in seedInvInfos)
  8073. {
  8074. current++;
  8075. // 计算进度:从 20% 到 90%
  8076. int progress = 20 + (int)((double)current / total * 70);
  8077. if (string.IsNullOrEmpty(item.EmailInfo?.EmailTitle) || string.IsNullOrEmpty(item.EmailInfo?.EmailContent))
  8078. {
  8079. await HttpContext.SendSseStepAsync(progress, $"跳过:{item.NameCn} (邮件标题或内容缺失)");
  8080. failCount++;
  8081. continue;
  8082. }
  8083. try
  8084. {
  8085. var req = new MailDto()
  8086. {
  8087. Subject = item.EmailInfo.EmailTitle,
  8088. Content = item.EmailInfo.EmailContent,
  8089. To = item.Email,
  8090. AttachmentPaths = item.EmailInfo.AttachmentPaths
  8091. };
  8092. var res = await _hotmailService.SendMailAsync(hotmailConfig.UserName, req);
  8093. if (res.IsSuccess)
  8094. {
  8095. successCount++;
  8096. // 更新本地状态
  8097. item.EmailInfo.Status = 4; // 发送成功状态
  8098. item.EmailInfo.Operator = userInfo.CnName;
  8099. item.EmailInfo.OperatedAt = DateTime.Now;
  8100. await HttpContext.SendSseStepAsync(progress, $"成功:已向 {item.NameCn}({item.Email}) 发送邮件");
  8101. }
  8102. else
  8103. {
  8104. failCount++;
  8105. await HttpContext.SendSseStepAsync(progress, $"失败:{item.NameCn} 发送失败 ({res.Message})");
  8106. }
  8107. }
  8108. catch (Exception ex)
  8109. {
  8110. failCount++;
  8111. _logger.LogError(ex, $"企微邮件推送异常:{item.NameCn}");
  8112. await HttpContext.SendSseStepAsync(progress, $"异常:{item.NameCn} 连接超时");
  8113. }
  8114. }
  8115. #endregion
  8116. await HttpContext.SendSseStepAsync(95, "正在归档发送记录...");
  8117. #region 3. 数据同步与收尾
  8118. // 更新内存中的全量数据:排除旧的,加入已更新状态的
  8119. opAiInfo.AiCrawledDetails = opAiInfo.AiCrawledDetails
  8120. .Where(x => !guidSet.Contains(x.Guid))
  8121. .Concat(seedInvInfos)
  8122. .OrderByDescending(x => x.EmailInfo.OperatedAt)
  8123. .ToList();
  8124. // 精准更新 JSON 列
  8125. var updateOk = await _sqlSugar.Updateable(opAiInfo)
  8126. .UpdateColumns(x => x.AiCrawledDetails)
  8127. .ExecuteCommandHasChangeAsync();
  8128. if (!updateOk)
  8129. {
  8130. await HttpContext.SendSseStepAsync(-1, "记录保存失败,请检查数据库连接。");
  8131. return;
  8132. }
  8133. await HttpContext.SendSseStepAsync(100, $"任务完成!成功: {successCount}, 失败: {failCount}", new
  8134. {
  8135. Id = opAiInfo.Id,
  8136. SuccessCount = successCount,
  8137. FailCount = failCount
  8138. });
  8139. #endregion
  8140. }
  8141. catch (Exception ex)
  8142. {
  8143. _logger.LogError(ex, "邮件发送流发生崩溃");
  8144. await HttpContext.SendSseStepAsync(-1, $"系统错误:{ex.Message}");
  8145. }
  8146. finally
  8147. {
  8148. await HttpContext.FinalizeSseAsync();
  8149. }
  8150. }
  8151. #endregion
  8152. }
  8153. }