| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675 | 'use strict';/** * * * @author Mai * @date 2018/8/14 * @version */const stdDataAddType = {    withParent: 1,    child: 2,    next: 3,};const moment = require('moment');const sendToWormhole = require('stream-wormhole');const fs = require('fs');const path = require('path');const audit = require('../const/audit');const codeRuleConst = require('../const/code_rule');const changeConst = require('../const/change');const accountGroup = require('../const/account_group').group;const shenpiConst = require('../const/shenpi');const tenderMenu = require('../../config/menu').tenderMenu;const measureType = require('../const/tender').measureType;const spreadConst = require('../const/spread');// const tenderMenu = require('../../config/menu').tenderMenu;module.exports = app => {    class ChangeController extends app.BaseController {        /**         * 构造函数         *         * @param {Object} ctx - egg全局变量         * @return {void}         */        constructor(ctx) {            super(ctx);            ctx.showProject = true;            ctx.showTender = true;            ctx.showTitle = true;        }        async _filterChanges(ctx, status = 0) {            const tenderId = ctx.params.id;            ctx.session.sessionUser.tenderId = tenderId;            const tender = await this.service.tender.getDataById(tenderId);            // const tenderList = await this.service.tender.getList();            const page = ctx.page;            const sorts = ctx.query.sort ? ctx.query.sort : 0;            const orders = ctx.query.order ? ctx.query.order : 0;            const changes = await ctx.service.change.getListByStatus(tender.id, status, 1, sorts, orders);            const total = await ctx.service.change.getCountByStatus(tender.id, status);            let page_total = 0;            const tp = await ctx.service.change.getTp(tender.id, status);            if (changes !== null) {                let i = 0;                for (const c of changes) {                    page_total = ctx.helper.add(page_total, c.total_price);                    const status = c.status === audit.flow.status.uncheck ? 0 : 1;                    // 根据审批人对当前变更令的状态取不同的展示方式。                    let changeAudit = '';                    let auditStatus = 0;                    switch (c.status) {                        case 1:                            auditStatus = 1;                            break;                        case 2:                            changeAudit = await ctx.service.changeAudit.getLastUser(c.cid, c.times, status);                            auditStatus = changeAudit.uid === ctx.session.sessionUser.accountId ? 1 : 0;                            break;                        case 3:                        case 4:                            auditStatus = 0;                            changeAudit = await ctx.service.changeAudit.getLastUser(c.cid, c.times, status);                            break;                        case 5:                            changeAudit = await ctx.service.changeAudit.getLastUser(c.cid, c.times - 1, status);                            auditStatus = c.uid === ctx.session.sessionUser.accountId ? 1 : 0;                            const back_changeUsedData = await ctx.service.stageChange.getFinalUsedData(ctx.tender.id, c.cid);                            c.stageChangeNum = this.ctx.helper.sum(back_changeUsedData.map(x => { return Math.abs(x.qty); }));                            break;                        case 6:                            changeAudit = await ctx.service.changeAudit.getLastBackUser(c.cid, c.times);                            const checkingAudit = await ctx.service.changeAudit.getLastUser(c.cid, c.times, status);                            auditStatus = checkingAudit.uid === ctx.session.sessionUser.accountId ? 1 : 0;                            break;                        case 9:                            auditStatus = 9;                            const changeUsedData = await ctx.service.stageChange.getFinalUsedData(ctx.tender.id, c.cid);                            c.stageChangeNum = this.ctx.helper.sum(changeUsedData.map(x => { return Math.abs(x.qty); }));                            break;                        default:                            break;                    }                    changes[i].changeAudit = changeAudit;                    changes[i].auditStatus = auditStatus;                    i++;                }            }            // 分页相关            const pageInfo = {                page,                total: Math.ceil(total / app.config.pageSize),                queryData: JSON.stringify(ctx.urlInfo.query),            };            const filter = JSON.parse(JSON.stringify(audit.filter));            filter.count = [];            filter.count[filter.status.pending] = await ctx.service.change.getCountByStatus(tender.id, filter.status.pending);// await ctx.service.change.pendingDatas(tender.id, ctx.session.sessionUser.accountId);            filter.count[filter.status.uncheck] = await ctx.service.change.getCountByStatus(tender.id, filter.status.uncheck);// await ctx.service.change.checkingDatas(tender.id, ctx.session.sessionUser.accountId);            filter.count[filter.status.checking] = await ctx.service.change.getCountByStatus(tender.id, filter.status.checking);// await ctx.service.change.checkedDatas(tender.id, ctx.session.sessionUser.accountId);            filter.count[filter.status.checked] = await ctx.service.change.getCountByStatus(tender.id, filter.status.checked);// await ctx.service.change.pendingDatas(tender.id, ctx.session.sessionUser.accountId);            filter.count[filter.status.checkNo] = await ctx.service.change.getCountByStatus(tender.id, filter.status.checkNo);// await ctx.service.change.pendingDatas(tender.id, ctx.session.sessionUser.accountId);            const codeRule = tender.c_rule ? JSON.parse(tender.c_rule) : [];            for (const rule of codeRule) {                switch (rule.rule_type) {                    case codeRuleConst.measure.ruleType.dealCode:                        rule.preview = ctx.tender.info.deal_info.dealCode;                        break;                    case codeRuleConst.measure.ruleType.tenderName:                        rule.preview = tender.name;                        break;                    case codeRuleConst.measure.ruleType.inDate:                        rule.preview = moment().format('YYYY');                        break;                    case codeRuleConst.measure.ruleType.text:                        rule.preview = rule.text;                        break;                    case codeRuleConst.measure.ruleType.addNo:                        const s = '0000000000';                        rule.preview = s.substr(s.length - rule.format);                        break;                    default: break;                }            }            const changePlanList = await ctx.service.changePlan.getAllDataByCondition({ where: { tid: tender.id, status: audit.changePlan.status.checked } });            const allPlanCodes = await ctx.service.change.getAllDataByCondition({                columns: ['plan_code'],                where: {                    tid: tender.id,                },            });            const apLists = allPlanCodes.length > 0 ? ctx.app._.uniq(ctx.app._.map(allPlanCodes, 'plan_code')) : [];            const renderData = {                uid: ctx.session.sessionUser.accountId,                moment,                tender,                // tenderList,                pageInfo,                changes,                page_total,                tp,                filter,                status,                codeRule,                dealCode: ctx.tender.info.deal_info.dealCode,                auditConst: audit.flow,                changeConst,                ruleType: codeRuleConst.ruleType.change,                ruleConst: codeRuleConst.measure,                tenderMenu: this.menu.tenderMenu,                preUrl: '/tender/' + tenderId,                tpUnit: ctx.tender.info.decimal.tp,                changePlanList,                apLists,            };            await this.layout('change/index.ejs', renderData, 'change/modal.ejs');        }        /**         * 变更管理 页面 (Get)         *         * @param {Object} ctx - egg全局变量         * @return {void}         */        async index(ctx) {            try {                await this._filterChanges(ctx);            } catch (err) {                this.log(err);                ctx.redirect('/dashboard');            }        }        /**         *         * @param {Object} ctx - egg全局变量         * @return {void}         */        async newCode(ctx) {            const responseData = {                err: 0,                msg: '',                data: '',            };            try {                const tenderId = ctx.params.id;                if (!tenderId) {                    throw '当前未打开标段';                }                const tenderData = await ctx.service.tender.getDataById(tenderId);                const data = JSON.parse(ctx.request.body.data);                let cCodeRule;                let cConnector;                let changeCount = 0;                if (data && data.type) {                    const c_code_rules = tenderData.c_code_rules !== null ? JSON.parse(tenderData.c_code_rules) : null;                    cCodeRule = c_code_rules && c_code_rules[data.type + '_rule'] ? c_code_rules[data.type + '_rule'] : [];                    cConnector = c_code_rules && c_code_rules[data.type + '_connector'] ? c_code_rules[data.type + '_connector'] : null;                    if (data.type === codeRuleConst.ruleField[codeRuleConst.ruleType.apply]) {                        changeCount = await ctx.service.changeApply.count({ tid: tenderId });                    } else {                        changeCount = await ctx.service.changeProject.count({ tid: tenderId, type: codeRuleConst.ruleType[data.type] });                    }                } else {                    cCodeRule = tenderData.c_rule !== null ? JSON.parse(tenderData.c_rule) : [];                    cConnector = tenderData.c_connector;                    changeCount = await ctx.service.change.count({ tid: tenderId });                }                const code = [];                for (const rule of cCodeRule) {                    switch (rule.rule_type) {                        case codeRuleConst.measure.ruleType.dealCode:                            code.push(ctx.tender.info.deal_info.dealCode);                            break;                        case codeRuleConst.measure.ruleType.tenderName:                            code.push(tenderData.name);                            break;                        case codeRuleConst.measure.ruleType.text:                            code.push(rule.text);                            break;                        case codeRuleConst.measure.ruleType.inDate:                            code.push(moment().format('YYYY'));                            break;                        case codeRuleConst.measure.ruleType.addNo:                            let s = '0000000000';                            const count = rule.start + changeCount;                            s = s + count;                            code.push(s.substr(s.length - rule.format));                            break;                        default: break;                    }                }                responseData.data = code.join(cConnector !== null && cConnector !== 3 ? codeRuleConst.measure.connectorString[cConnector] : '');            } catch (err) {                responseData.err = 1;                responseData.msg = err;            }            ctx.body = responseData;        }        /**         * 新增变更 (Post)         *         * @param {Object} ctx - egg全局变量         * @return {void}         */        async add(ctx) {            try {                const tenderId = ctx.params.id;                if (!tenderId) {                    throw '当前未打开标段';                }                const data = JSON.parse(ctx.request.body.data);                if (!data.code || data.code === '') {                    throw '变更令号不能为空';                }                const change = await ctx.service.change.add(tenderId, ctx.session.sessionUser.accountId, data.code, data.plan_code, data.name);                ctx.body = { err: 0, msg: '', data: change };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString() };            }        }        /**         * 变更管理 状态筛选 页面 (Get)         * @param {Object} ctx - egg全局变量         * @return {void}         */        async status(ctx) {            try {                const status = parseInt(ctx.params.status);                await this._filterChanges(ctx, status);            } catch (err) {                this.logger.error(err);                ctx.redirect('/tender/' + ctx.params.id + '/change');            }        }        /**         * 变更信息 页面 (Get)         *         * @param {Object} ctx - egg全局变量         * @return {void}         */        async info(ctx) {            try {                const whiteList = this.ctx.app.config.multipart.whitelist;                const tenderid = ctx.params.id !== undefined ? ctx.params.id : ctx.session.sessionUser.tenderId;                ctx.session.sessionUser.tenderId = tenderid;                const tender = await ctx.service.tender.getDataById(tenderid);                const change = await ctx.service.change.getDataByCondition({ cid: ctx.params.cid });                // 后台判断当前人查看info状态                const auditStatus = await ctx.service.changeAudit.getStatusByChange(change);                // 获取附件列表                const attList = await ctx.service.changeAtt.getChangeAttachment(ctx.params.cid);                // 获取其他变更令数据                const othersChange = await ctx.service.change.getOthersChange(ctx.params.id, ctx.params.cid);                // 根据auditStatus获取审批人列表                const auditList = await ctx.service.changeAudit.getListByStatus(change, auditStatus);                // 获取已选清单                let changeList = await ctx.service.changeAuditList.getAllDataByCondition({ where: { cid: ctx.params.cid } });                // 获取用户人验证手机号                const pa = await ctx.service.projectAccount.getDataById(ctx.session.sessionUser.accountId);                const auth_mobile = pa.auth_mobile;                const renderData = {                    uid: ctx.session.sessionUser.accountId,                    tender,                    change,                    othersChange,                    changeConst,                    auditStatus,                    auditConst: audit.flow,                    ledgerConsts: audit.ledger.status,                    attList,                    whiteList,                    auditList,                    changeList,                    tpUnit: change.tp_decimal !== null ? change.tp_decimal : ctx.tender.info.decimal.tp,                    upUnit: change.up_decimal !== null ? change.up_decimal : ctx.tender.info.decimal.up,                    authMobile: auth_mobile,                    shenpiConst,                };                // 根据auditStatus状态获取的不同的数据                if (auditStatus === 1 || auditStatus === 2 || auditStatus === 9) {                    renderData.changeUnits = changeConst.units;                    renderData.precision = ctx.tender.info.precision;                    // renderData.accountGroup = accountGroup;                    // 获取所有项目参与者                    const accountList = await ctx.service.projectAccount.getAllDataByCondition({                        where: { project_id: ctx.session.sessionProject.id, enable: 1 },                        columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group'],                    });                    renderData.accountList = accountList;                    renderData.accountGroup = accountGroup.map((item, idx) => {                        const groupList = accountList.filter(item => item.account_group === idx);                        return { groupName: item, groupList };                    });                    // 重新上报获取审批流程                    if (auditStatus === 2 || auditStatus === 9) {                        const auditList2 = await ctx.service.changeAudit.getListByBack(change.cid, change.times);                        // 展示页右侧审批流程列表                        const auditList3 = [];                        for (let time = 1; time <= change.times - 1; time++) {                            const auditTimeList = [];                            let max_sort = 1;                            for (const al of auditList2) {                                if (al.times === time) {                                    auditTimeList.push(al);                                    if (al.usite > max_sort) {                                        max_sort = al.usite;                                    }                                }                            }                            for (const i in auditTimeList) {                                auditTimeList[i].max_sort = max_sort;                            }                            auditList3.push(auditTimeList);                        }                        renderData.auditList3 = auditList3;                    }                    // 根据清单获取提交数据和计算总金额                    const changeListData = [];                    const changeWhiteListData = [];                    let ototalCost = 0;                    let ctotalCost = 0;                    for (const cl of changeList) {                        const cLArray = [                            cl.code,                            cl.name,                            cl.bwmx,                            cl.unit,                            cl.unit_price,                            cl.oamount,                            cl.camount,                            cl.detail,                            cl.lid,                            cl.xmj_code,                            cl.xmj_jldy,                            cl.gcl_id,                        ];                        ototalCost += cl.unit_price === null ? 0 : ctx.helper.mul(cl.unit_price, cl.oamount, ctx.tender.info.decimal.tp);                        ctotalCost += cl.unit_price === null ? 0 : ctx.helper.mul(cl.unit_price, cl.camount, ctx.tender.info.decimal.tp);                        if (cl.lid !== '0') {                            changeListData.push(cLArray.join('*;*'));                        } else {                            changeWhiteListData.push(cLArray.join('*;*'));                        }                    }                    renderData.changeListData = changeListData.join('^_^');                    renderData.changeWhiteListData = changeWhiteListData.join('^_^');                    renderData.ototalCost = ototalCost;                    renderData.ctotalCost = ctotalCost;                    // 获取公司列表                    const companyList = await ctx.service.changeCompany.getAllDataByCondition({ where: { tid: tenderid } });                    renderData.companyList = companyList;                } else if (auditStatus === 3 || auditStatus === 4 || auditStatus === 5 || auditStatus === 7) {                    // 展示页左侧审批流程列表和清单审批列表数据                    const times = change.status === audit.flow.status.back ?                        change.times - 1 : change.times;                    const auditList2 = await ctx.service.changeAudit.getListGroupByTimes(change.cid, times);                    // 展示页右侧审批流程列表                    const auditList3 = [];                    for (let time = 1; time <= times; time++) {                        const auditTimeList = [];                        let max_sort = 1;                        for (const al of auditList) {                            if (al.times === time) {                                auditTimeList.push(al);                                if (al.usite > max_sort) {                                    max_sort = al.usite;                                }                            }                        }                        for (const i in auditTimeList) {                            auditTimeList[i].max_sort = max_sort;                        }                        auditList3.push(auditTimeList);                    }                    renderData.auditList3 = auditList3;                    changeList = JSON.parse(JSON.stringify(changeList.sort())).sort().sort();                    renderData.changeList = changeList;                    let ototalCost = 0;                    let ctotalCost = 0;                    let stotalCost = 0;                    const auditTotalCost = [];                    for (const cl of changeList) {                        // ototalCost += cl.unit_price === null ? 0 : parseFloat(ctx.helper.roundNum(ctx.helper.accMul(cl.unit_price, cl.oamount), renderData.tpUnit));                        ototalCost += cl.unit_price === null ? 0 : ctx.helper.mul(cl.unit_price, cl.oamount, renderData.tpUnit);                        // ctotalCost += cl.unit_price === null ? 0 : parseFloat(ctx.helper.roundNum(ctx.helper.accMul(cl.unit_price, cl.camount), renderData.tpUnit));                        ctotalCost += cl.unit_price === null ? 0 : ctx.helper.mul(cl.unit_price, cl.camount, renderData.tpUnit);                        // stotalCost += cl.samount !== '' && cl.unit_price !== null ? parseFloat(ctx.helper.roundNum(ctx.helper.accMul(cl.unit_price, cl.samount), renderData.tpUnit)) : 0;                        stotalCost += cl.samount !== '' && cl.unit_price !== null ? ctx.helper.mul(cl.unit_price, cl.samount, renderData.tpUnit) : 0;                        const audit_amount = cl.audit_amount !== null && cl.audit_amount !== '' ? cl.audit_amount.split(',') : '';                        auditTotalCost.push(audit_amount);                    }                    renderData.ototalCost = ototalCost;                    renderData.ctotalCost = ctotalCost;                    renderData.stotalCost = stotalCost;                    // 清单表页赋值                    for (const [index, au] of auditList2.entries()) {                        if (au.usite !== 0) {                            au.list_amount = [];                            au.totalCost = 0;                            for (const [auindex, at] of auditTotalCost.entries()) {                                au.list_amount.push(at[index - 1]);                                // au.totalCost += at[index - 1] !== undefined && changeList[auindex].unit_price !== null ? parseFloat(ctx.helper.roundNum(ctx.helper.accMul(changeList[auindex].unit_price, at[index - 1]), renderData.tpUnit)) : 0;                                au.totalCost += at[index - 1] !== undefined && changeList[auindex].unit_price !== null ? ctx.helper.mul(changeList[auindex].unit_price, at[index - 1], renderData.tpUnit) : 0;                            }                        }                    }                    renderData.auditList2 = auditList2;                } else if (auditStatus === 6) {                    // 展示页左侧审批流程列表和清单审批列表数据                    const auditList2 = await ctx.service.changeAudit.getListGroupByTimes(change.cid, change.times);                    renderData.auditList2 = auditList2;                    const auditList3 = await ctx.service.changeAudit.getListOrderByTimes(change.cid, change.times);                    for (const i in auditList3) {                        auditList3[i].max_sort = auditList2.length - 1;                    }                    renderData.auditList3 = auditList3;                    // 展示页右侧审批流程列表                    const auditList5 = await ctx.service.changeAudit.getListByBack(change.cid, change.times);                    const auditList4 = [];                    for (let time = 1; time <= change.times; time++) {                        const auditTimeList = [];                        let max_sort = 1;                        for (const al of auditList5) {                            if (al.times === time) {                                auditTimeList.push(al);                                if (al.usite > max_sort) {                                    max_sort = al.usite;                                }                            }                        }                        for (const i in auditTimeList) {                            auditTimeList[i].max_sort = max_sort;                        }                        if (auditTimeList.length > 0) {                            auditList4.push(auditTimeList);                        }                    }                    renderData.auditList4 = auditList4;                    changeList = JSON.parse(JSON.stringify(changeList.sort())).sort().sort();                    renderData.changeList = changeList;                    let ototalCost = 0;                    let ctotalCost = 0;                    const auditTotalCost = [];                    const auditUnit = [];                    for (const cl of changeList) {                        // ototalCost += cl.unit_price === null ? 0 : parseFloat(ctx.helper.roundNum(ctx.helper.accMul(cl.unit_price, cl.oamount), renderData.tpUnit));                        ototalCost += cl.unit_price === null ? 0 : ctx.helper.mul(cl.unit_price, cl.oamount, renderData.tpUnit);                        // ctotalCost += cl.unit_price === null ? 0 : parseFloat(ctx.helper.roundNum(ctx.helper.accMul(cl.unit_price, cl.camount), renderData.tpUnit));                        ctotalCost += cl.unit_price === null ? 0 : ctx.helper.mul(cl.unit_price, cl.camount, renderData.tpUnit);                        const audit_amount = cl.audit_amount !== null && cl.audit_amount !== '' ? cl.audit_amount.split(',') : '';                        auditTotalCost.push(audit_amount);                    }                    renderData.ototalCost = ototalCost;                    renderData.ctotalCost = ctotalCost;                    // 清单表页赋值                    for (const [index, au] of auditList.entries()) {                        if (au.usite !== 0) {                            au.list_amount = [];                            au.totalCost = 0;                            if (au.uid === renderData.uid) {                                for (const [auindex, at] of auditTotalCost.entries()) {                                    // if (at[index - 2] !== undefined) {                                    //     au.list_amount.push(at[index - 2]);                                    //     au.totalCost += parseFloat(ctx.helper.roundNum(ctx.helper.accMul(changeList[auindex].unit_price, at[index - 2]), renderData.tpUnit));                                    // } else if (at[index - 2] === undefined) {                                    //     au.list_amount.push(changeList[auindex].camount);                                    //     au.totalCost += parseFloat(ctx.helper.roundNum(ctx.helper.accMul(changeList[auindex].unit_price, changeList[auindex].camount), renderData.tpUnit));                                    // }                                    au.list_amount.push(changeList[auindex].spamount);                                    // au.totalCost += changeList[auindex].unit_price === null ? 0 : parseFloat(ctx.helper.roundNum(ctx.helper.accMul(changeList[auindex].unit_price, changeList[auindex].spamount), renderData.tpUnit));                                    au.totalCost += changeList[auindex].unit_price === null ? 0 : ctx.helper.mul(changeList[auindex].unit_price, changeList[auindex].spamount, renderData.tpUnit);                                }                            } else {                                for (const [auindex, at] of auditTotalCost.entries()) {                                    au.list_amount.push(at[index - 1]);                                    // au.totalCost += at[index - 1] !== undefined && changeList[auindex].unit_price !== null ? parseFloat(ctx.helper.roundNum(ctx.helper.accMul(changeList[auindex].unit_price, at[index - 1]), renderData.tpUnit)) : 0;                                    au.totalCost += at[index - 1] !== undefined && changeList[auindex].unit_price !== null ? ctx.helper.mul(changeList[auindex].unit_price, at[index - 1], renderData.tpUnit) : 0;                                }                            }                        }                    }                }                renderData.auditList = auditList;                // 获取是否已存在调用变更令                const changeUsedData = await ctx.service.stageChange.getFinalUsedData(ctx.tender.id, change.cid);                renderData.stageChangeNum = this.ctx.helper.sum(changeUsedData.map(x => { return Math.abs(x.qty); }));                await this.layout('change/info.ejs', renderData, 'change/info_modal.ejs');            } catch (err) {                this.log(err);                ctx.redirect('/tender/' + ctx.params.id + '/change');            }        }        /**         * 变更信息 新版本页面 (Get)         *         * @param {Object} ctx - egg全局变量         * @return {void}         */        async information(ctx) {            try {                const whiteList = this.ctx.app.config.multipart.whitelist;                const tender = await ctx.service.tender.getDataById(ctx.tender.id);                const change = ctx.change;                // 后台判断当前人查看info状态                const auditStatus = await ctx.service.changeAudit.getStatusByChange(change);                // 获取附件列表                const attList = await ctx.service.changeAtt.getChangeAttachment(change.cid, 'desc');                // 获取其他变更令数据                const othersChange = await ctx.service.change.getOthersChange(ctx.tender.id, change.cid);                // 根据auditStatus获取审批人列表                const auditList = await ctx.service.changeAudit.getListByStatus(change, auditStatus);                // 获取变更方案的清单                let planList = [];                let showPlanBtn = false;                if (ctx.session.sessionProject.page_show.openChangePlan) {                    const planInfo = change.plan_code ? await ctx.service.changePlan.getDataByCondition({ tid: ctx.tender.id, code: change.plan_code }) : null;                    showPlanBtn = change.plan_code !== null && change.plan_code !== '';                    if (planInfo && planInfo.id) {                        planList = await ctx.service.changePlanList.getAllDataByCondition({ where: { cpid: planInfo.id } });                    }                }                // 获取用户人验证手机号                const pa = await ctx.service.projectAccount.getDataById(ctx.session.sessionUser.accountId);                const auth_mobile = pa.auth_mobile;                const renderData = {                    tender,                    change,                    othersChange,                    changeConst,                    auditStatus,                    auditConst: audit.flow,                    ledgerConsts: audit.ledger.status,                    attList,                    whiteList,                    auditList,                    showPlanBtn,                    planList,                    tpUnit: change.tp_decimal ? change.tp_decimal : ctx.tender.info.decimal.tp,                    upUnit: change.up_decimal ? change.up_decimal : ctx.tender.info.decimal.up,                    authMobile: auth_mobile,                    shenpiConst,                    jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.change.information),                    preUrl: '/tender/' + ctx.tender.id + '/change/' + ctx.change.cid + '/information',                    precision: ctx.tender.info.precision,                };                // 根据auditStatus状态获取的不同的数据                if (auditStatus === 1 || auditStatus === 2 || auditStatus === 9) {                    renderData.changeUnits = changeConst.units;                    // renderData.accountGroup = accountGroup;                    // 获取所有项目参与者                    const accountList = await ctx.service.projectAccount.getAllDataByCondition({                        where: { project_id: ctx.session.sessionProject.id, enable: 1 },                        columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group'],                    });                    renderData.accountList = accountList;                    renderData.accountGroup = accountGroup.map((item, idx) => {                        const groupList = accountList.filter(item => item.account_group === idx);                        return { groupName: item, groupList };                    });                    // 重新上报获取审批流程                    if (auditStatus === 2 || auditStatus === 9) {                        const auditList2 = await ctx.service.changeAudit.getListByBack(change.cid, change.times);                        // 展示页右侧审批流程列表                        const auditList3 = [];                        for (let time = 1; time <= change.times - 1; time++) {                            const auditTimeList = [];                            let max_sort = 1;                            for (const al of auditList2) {                                if (al.times === time) {                                    auditTimeList.push(al);                                    if (al.usite > max_sort) {                                        max_sort = al.usite;                                    }                                }                            }                            for (const i in auditTimeList) {                                auditTimeList[i].max_sort = max_sort;                            }                            auditList3.push(auditTimeList);                        }                        renderData.auditList3 = auditList3;                    }                    // 获取公司列表                    const companyList = await ctx.service.changeCompany.getAllDataByCondition({ where: { tid: ctx.tender.id } });                    renderData.companyList = companyList;                    // 获取已选清单                    const changeList = await ctx.service.changeAuditList.getList(change.cid);                    renderData.changeList = changeList;                    renderData.changeLedgerList = await ctx.service.changeLedger.getAllDataByCondition({ where: { tender_id: ctx.tender.id } });                    renderData.changePosList = await ctx.service.changePos.getAllDataByCondition({ where: { tid: ctx.tender.id } });                } else if (auditStatus === 3 || auditStatus === 4 || auditStatus === 5 || auditStatus === 7 || auditStatus === 8) {                    // 展示页左侧审批流程列表和清单审批列表数据                    const times = change.status === audit.flow.status.back ?                        change.times - 1 : change.times;                    const auditList2 = await ctx.service.changeAudit.getListGroupByTimes(change.cid, times);                    // 展示页右侧审批流程列表                    const auditList3 = [];                    for (let time = 1; time <= times; time++) {                        const auditTimeList = [];                        let max_sort = 1;                        for (const al of auditList) {                            if (al.times === time) {                                auditTimeList.push(al);                                if (al.usite > max_sort) {                                    max_sort = al.usite;                                }                            }                        }                        for (const i in auditTimeList) {                            auditTimeList[i].max_sort = max_sort;                        }                        auditList3.push(auditTimeList);                    }                    renderData.auditList3 = auditList3;                    // 获取已选清单                    // let changeList = await ctx.service.changeAuditList.getAllDataByCondition({ where: { cid: ctx.params.cid } });                    const changeList = await ctx.service.changeAuditList.getList(change.cid);                    // changeList = JSON.parse(JSON.stringify(changeList.sort())).sort().sort();                    for (const cl of changeList) {                        const audit_amount = cl.audit_amount !== null && cl.audit_amount !== '' ? cl.audit_amount.split(',') : '';                        // 清单表页赋值                        for (const [index, au] of auditList2.entries()) {                            if (au.usite !== 0) {                                cl['audit_amount_' + au.uid] = audit_amount[index - 1] ? audit_amount[index - 1] : null;                            }                        }                    }                    renderData.changeList = changeList;                    renderData.auditList2 = auditList2;                } else if (auditStatus === 6) {                    // 展示页左侧审批流程列表和清单审批列表数据                    const auditList2 = await ctx.service.changeAudit.getListGroupByTimes(change.cid, change.times);                    renderData.auditList2 = auditList2;                    const auditList3 = await ctx.service.changeAudit.getListOrderByTimes(change.cid, change.times);                    for (const i in auditList3) {                        auditList3[i].max_sort = auditList2.length - 1;                    }                    renderData.auditList3 = auditList3;                    // 展示页右侧审批流程列表                    const auditList5 = await ctx.service.changeAudit.getListByBack(change.cid, change.times);                    const auditList4 = [];                    for (let time = 1; time <= change.times; time++) {                        const auditTimeList = [];                        let max_sort = 1;                        for (const al of auditList5) {                            if (al.times === time) {                                auditTimeList.push(al);                                if (al.usite > max_sort) {                                    max_sort = al.usite;                                }                            }                        }                        for (const i in auditTimeList) {                            auditTimeList[i].max_sort = max_sort;                        }                        if (auditTimeList.length > 0) {                            auditList4.push(auditTimeList);                        }                    }                    renderData.auditList4 = auditList4;                    // 获取已选清单                    // let changeList = await ctx.service.changeAuditList.getAllDataByCondition({ where: { cid: ctx.params.cid } });                    const changeList = await ctx.service.changeAuditList.getList(change.cid);                    // changeList = JSON.parse(JSON.stringify(changeList.sort())).sort().sort();                    for (const cl of changeList) {                        const audit_amount = cl.audit_amount !== null && cl.audit_amount !== '' ? cl.audit_amount.split(',') : '';                        // 清单表页赋值                        for (const [index, au] of auditList2.entries()) {                            if (au.usite !== 0) {                                cl['audit_amount_' + au.uid] = au.uid === ctx.session.sessionUser.accountId ? cl.spamount : (audit_amount[index - 1] ? audit_amount[index - 1] : null);                            }                        }                    }                    renderData.changeList = changeList;                    renderData.changeLedgerList = await ctx.service.changeLedger.getAllDataByCondition({ where: { tender_id: ctx.tender.id } });                    renderData.changePosList = await ctx.service.changePos.getAllDataByCondition({ where: { tid: ctx.tender.id } });                }                renderData.auditList = auditList;                // 获取是否已存在调用变更令                let changeUsedData = await ctx.service.stageChange.getFinalUsedData(ctx.tender.id, change.cid);                changeUsedData = ctx.helper._.filter(changeUsedData, function(item) {                    return item.qty !== null;                })                renderData.changeUsedData = changeUsedData;                renderData.stageChangeNum = this.ctx.helper.sum(changeUsedData.map(x => { return Math.abs(x.qty); }));                await this.layout('change/information.ejs', renderData, 'change/information_modal.ejs');            } catch (err) {                this.log(err);                ctx.redirect('/tender/' + ctx.params.id + '/change');            }        }        /**         * 变更清单 - 操作 (Ajax)         * @param ctx         * @return {Promise<void>}         */        async saveListsData(ctx) {            try {                const data = JSON.parse(ctx.request.body.data);                const responseData = {                    err: 0,                    msg: '',                    data: {},                };                switch (data.type) {                    case 'add':                        responseData.data = await ctx.service.changeAuditList.add(data.postData);                        break;                    case 'batchadd':                        responseData.data = await ctx.service.changeAuditList.batchAdd(data);                        break;                    case 'del':                        await ctx.service.changeAuditList.del(data);                        break;                    case 'update':                        await ctx.service.changeAuditList.save(data.updateData);                        break;                    case 'paste':                        await ctx.service.changeAuditList.saveDatas(data.updateData);                        // 取所有工料表                        responseData.data = await ctx.service.changeAuditList.getList(ctx.change.cid);                        break;                    case 'ledger_list':                        await ctx.service.changeAuditList.saveLedgerListDatas(data.updateData, data.postData);                        // 取所有工料表                        responseData.data = { changeList: await ctx.service.changeAuditList.getList(ctx.change.cid),                            usedList: await ctx.service.stageChange.getFinalUsedData(ctx.tender.id, ctx.change.cid) };                        break;                    case 'remove_list':                        await ctx.service.changeAuditList.removeLedgerListDatas(data.updateData);                        // 取所有工料表                        responseData.data = await ctx.service.changeAuditList.getList(ctx.change.cid);                        break;                    case 'update_list':                        await ctx.service.changeAuditList.saveDatas(data.updateData);                        // 取所有工料表                        // responseData.data = await ctx.service.changeAuditList.getList(ctx.change.cid);                        break;                    case 'info':                        await ctx.service.change.saveInfo(data.updateData);                        // 取所有工料表                        responseData.data = '保存成功';                        break;                    case 'paste_amount_rows':                        await ctx.service.changeAuditList.saveDatas(data.updateData);                        const changeList = await ctx.service.changeAuditList.getList(ctx.change.cid);                        const auditList2 = await ctx.service.changeAudit.getListGroupByTimes(ctx.change.cid, ctx.change.times);                        // changeList = JSON.parse(JSON.stringify(changeList.sort())).sort().sort();                        for (const cl of changeList) {                            const audit_amount = cl.audit_amount !== null && cl.audit_amount !== '' ? cl.audit_amount.split(',') : '';                            // 清单表页赋值                            for (const [index, au] of auditList2.entries()) {                                if (au.usite !== 0) {                                    cl['audit_amount_' + au.uid] = au.uid === ctx.session.sessionUser.accountId ? cl.spamount : (audit_amount[index - 1] ? audit_amount[index - 1] : null);                                }                            }                        }                        // 取所有工料表                        responseData.data = changeList;                        break;                    case 'update_tp':                        await ctx.service.change.saveInfo({ total_price: data.updateData });                        break;                    case 'order_by':                        const result = await ctx.service.change.saveOrderBy(data.updateData, data.newLedgerList);                        responseData.data = result;                        break;                    case 'changeOrder':                        await ctx.service.changeAuditList.changeOrder(data.postData);                        break;                    default: throw '参数有误';                }                ctx.body = responseData;            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 上报和重新上报         * @param ctx         * @return {Promise<void>}         */        async startAudit(ctx) {            try {                // 检查权限等                if (!ctx.change) {                    throw '数据错误';                }                if (ctx.change.uid !== ctx.session.sessionUser.accountId) {                    throw '您无权上报该变更令数据';                }                if (!(ctx.change.status === audit.flow.status.uncheck || ctx.change.status === audit.flow.status.back || ctx.change.status === audit.flow.status.revise)) {                    throw '该变更令当前无法上报';                }                // 判断是否是修订,判断有无审批人员作弊                if ((ctx.change.status === audit.flow.status.revise || ctx.change.is_revise) && ctx.tender.info.shenpi.change === shenpiConst.sp_status.sqspr) {                    // 获取上一次的审批人流程                    const lastUserList = await ctx.service.changeAudit.getListGroupByTimes(ctx.change.cid, ctx.change.times - 1);                    const lastUidList = ctx.helper._.map(lastUserList, 'uid');                    // 判断上一次审批是否为非原报为审批人                    const nowUidList = ctx.helper._.map(ctx.change.auditors, 'uid');                    // 判断条件修订可上报条件                    // 1.有原报,有其他人   可以添加原报作为审核人,但是不能只原报上报                    // 2.有原报,无其他人   可以添加原报作为审核人,可以只原报上报                    // 3.无原报,有其他人   不可以添加原报                    const noYBUidList = ctx.change.status === audit.flow.status.revise ? ctx.helper._.initial(ctx.helper._.tail(lastUidList)) : ctx.helper._.tail(lastUidList);                    if (!ctx.helper._.isEqual(lastUidList, nowUidList) && ctx.helper._.includes(noYBUidList, lastUidList[0]) && nowUidList.length === 2 && nowUidList[0] === nowUidList[1]) {                        throw '该变更令不能指定原报人为单独审批人';                    }                    if (!ctx.helper._.isEqual(lastUidList, nowUidList) && !ctx.helper._.includes(noYBUidList, lastUidList[0]) && ctx.helper._.includes(ctx.helper._.tail(nowUidList), nowUidList[0])) {                        throw '该变更令不能指定原报为审批人,请移除';                    }                }                await ctx.service.changeAudit.start(ctx.change.cid, ctx.change.times);                ctx.redirect(ctx.request.header.referer);            } catch (err) {                this.log(err);                ctx.session.postError = err.toString();                ctx.redirect(ctx.request.header.referer);            }        }        // 审批相关        /**         * 添加审批人         * @param ctx         * @return {Promise<void>}         */        async addAudit(ctx) {            try {                const data = JSON.parse(ctx.request.body.data);                const id = this.app._.toInteger(data.auditorId);                if (isNaN(id) || id <= 0) {                    throw '参数错误';                }                // 检查权限等                if (ctx.change.uid !== ctx.session.sessionUser.accountId) {                    throw '您无权添加审核人';                }                if (ctx.change.status === audit.flow.status.checking || ctx.change.status === audit.flow.status.checked) {                    throw '当前不允许添加审核人';                }                const auditorList = await ctx.service.changeAudit.getAuditors(ctx.change.cid, ctx.change.times);                // 检查审核人是否已存在                const exist = this.app._.find(auditorList, { uid: id });                if (exist) {                    throw '该审核人已存在,请勿重复添加';                }                const shenpiInfo = await ctx.service.shenpiAudit.getDataByCondition({ tid: ctx.tender.id, sp_type: shenpiConst.sp_type.change, sp_status: shenpiConst.sp_status.gdzs });                const is_gdzs = shenpiInfo && ctx.tender.info.shenpi.change === shenpiConst.sp_status.gdzs ? 1 : 0;                const result = await ctx.service.changeAudit.addAuditor(ctx.change.cid, id, ctx.change.times, is_gdzs);                if (!result) {                    throw '添加审核人失败';                }                // const auditors = await ctx.service.changeAudit.getAuditorsWithOwner(ctx.change.id, ctx.change.times);                const auditors = null;                ctx.body = { err: 0, msg: '', data: auditors };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 移除审批人         * @param ctx         * @return {Promise<void>}         */        async deleteAudit(ctx) {            try {                const data = JSON.parse(ctx.request.body.data);                const id = data.auditorId instanceof Number ? data.auditorId : this.app._.toNumber(data.auditorId);                if (isNaN(id) || id <= 0) {                    throw '参数错误';                }                const result = await ctx.service.changeAudit.deleteAuditor(ctx.change.cid, id, ctx.change.times);                if (!result) {                    throw '移除审核人失败';                }                const auditors = null;                ctx.body = { err: 0, msg: '', data: auditors };            } catch (err) {                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        async defaultBills(ctx) {            try {                // 判断是否台账修订中,修订中则不获取changeLedger及changePos值                const lastRevise = await ctx.service.ledgerRevise.getLastestRevise(ctx.tender.id);                const data = JSON.parse(ctx.request.body.data);                const ledgerData = await ctx.service.ledger.getData(ctx.tender.id);                const changeLedgerData = !ctx.session.sessionProject.page_show.openChangeRevise ? [] : (lastRevise && lastRevise.status !== audit.revise.status.checked && data.from !== 'revise' ? [] : await ctx.service.changeLedger.getData(ctx.tender.id));                const posData = await ctx.service.pos.getPosData({ tid: ctx.tender.id });                const changePosData = !ctx.session.sessionProject.page_show.openChangeRevise ? [] : (lastRevise && lastRevise.status !== audit.revise.status.checked && data.from !== 'revise' ? [] : await ctx.service.changePos.getPosData({ tid: ctx.tender.id }));                const dealBills = await ctx.service.dealBills.getAllDataByCondition({ where: { tender_id: ctx.tender.id } });                // 标记ledger,搜索需求                if (changePosData.length > 0) {                    const cplIdList = ctx.helper._.uniq(ctx.helper._.map(changePosData, 'lid'));                    const cLIdList = ctx.helper._.map(changeLedgerData, 'id');                    const ledgerIdList = [];                    for (const cp of cplIdList) {                        if (ctx.helper._.indexOf(cLIdList, cp) === -1) {                            ledgerIdList.push(cp);                        }                    }                    if (ledgerIdList.length > 0) {                        for (const lid of ledgerIdList) {                            const data = ctx.helper._.find(ledgerData, { id: lid });                            data.cid = 1;                        }                    }                }                ctx.body = { err: 0, msg: '', data: { bills: ctx.helper._.concat(ledgerData, changeLedgerData), pos: ctx.helper._.concat(posData, changePosData), dealBills } };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: [] };            }        }        /**         * 变更令上报和保存         * @param {Object} ctx - egg全局变量         * @return {void}         */        async save(ctx) {            // 变更令信息            const changeInfo = await ctx.service.change.getDataByCondition({ cid: ctx.request.body.cid });            try {                const result = await ctx.service.change.save(ctx.request.body, changeInfo.tid);                if (!result) {                    throw '上报失败';                }                if (ctx.request.body.changestatus !== undefined && parseInt(ctx.request.body.changestatus) === 2) {                    ctx.body = { err: 0, msg: '保存成功' };                } else {                    ctx.redirect('/tender/' + changeInfo.tid + '/change');                }            } catch (err) {                this.log(err);                if (ctx.request.body.changestatus !== undefined && parseInt(ctx.request.body.changestatus) === 2) {                    ctx.body = { err: 1, msg: err.toString() };                } else {                    ctx.redirect('/tender/' + changeInfo.tid + '/change/' + ctx.request.body.cid + '/info');                }            }        }        /**         * 变更令审批         * @param {Object} ctx - egg全局变量         * @return {void}         */        async approval(ctx) {            try {                const changeData = await ctx.service.change.getDataByCondition({ cid: ctx.request.body.change_id });                if (!changeData) {                    throw '变更令数据错误';                }                const status = parseInt(ctx.request.body.status);                let result = false;                const pid = this.ctx.session.sessionProject.id;                switch (status) {                    case 3:// 审批通过                        result = await ctx.service.change.approvalSuccess(pid, ctx.request.body, changeData);                        break;                    case 4:// 审批终止                        result = await ctx.service.change.approvalStop(ctx.request.body);                        break;                    case 5:// 审批退回到原报人                        result = await ctx.service.change.approvalBack(pid, ctx.request.body, changeData);                        break;                    case 6:// 审批退回到上一个审批人                        result = await ctx.service.change.approvalBackNew(pid, ctx.request.body, changeData);                        break;                    default:break;                }                if (!result) {                    throw '审批失败';                }                ctx.redirect(ctx.request.header.referer);            } catch (err) {                console.log(err);                ctx.redirect(ctx.request.header.referer);            }        }        /**         * 变更公司管理         * @param {Object} ctx - egg全局变量         * @return {void}         */        async updateCompany(ctx) {            const responseData = {                err: 0,                msg: '',                data: '',            };            try {                const data = JSON.parse(ctx.request.body.data);                if (data.tid === undefined || data.uci === undefined || data.uc === undefined || data.ac === undefined) {                    throw '参数有误';                }                const [addCompany, selectCompany] = await ctx.service.changeCompany.setCompanyList(data);                responseData.data = { add: addCompany, select: selectCompany };            } catch (err) {                responseData.err = 1;                responseData.msg = err;            }            ctx.body = responseData;        }        /**         * 上传附件         * @param {Object} ctx - egg全局变量         * @return {void}         */        async uploadFile(ctx) {            const responseData = {                err: 0,                msg: '',                data: [],            };            let stream;            try {                const parts = ctx.multipart({ autoFields: true });                const files = [];                let index = 0;                const change = await ctx.service.change.getDataByCondition({ cid: ctx.params.cid });                const extra_upload = change.status === audit.flow.status.checked;                while ((stream = await parts()) !== undefined) {                    // 判断用户是否选择上传文件                    if (!stream.filename) {                        throw '请选择上传的文件!';                    }                    // const create_time = Date.parse(new Date()) / 1000;                    // const fileInfo = path.parse(stream.filename);                    // const dirName = 'app/public/upload/changes/' + moment().format('YYYYMMDD');                    // const fileName = 'changes' + create_time + '_' + index + fileInfo.ext;                    // // 判断文件夹是否存在,不存在则直接创建文件夹                    // if (!fs.existsSync(path.join(this.app.baseDir, dirName))) {                    //     await fs.mkdirSync(path.join(this.app.baseDir, dirName));                    // }                    // // 保存文件                    // await ctx.helper.saveStreamFile(stream, path.join(this.app.baseDir, dirName, fileName));                    const fileInfo = path.parse(stream.filename);                    const create_time = Date.parse(new Date()) / 1000;                    // const filepath = `app/public/upload/change/fujian_${create_time + index.toString() + fileInfo.ext}`;                    const filepath = `app/public/upload/${parts.field.tid}/change/fujian_${create_time + index.toString() + fileInfo.ext}`;                    // await ctx.helper.saveStreamFile(stream, path.resolve(this.app.baseDir, filepath));                    await ctx.app.fujianOss.put(ctx.app.config.fujianOssFolder + filepath, stream);                    await sendToWormhole(stream);                    // 保存数据到att表                    const fileData = {                        in_time: create_time,                        filename: fileInfo.name,                        fileext: fileInfo.ext,                        filesize: Array.isArray(parts.field.size) ? parts.field.size[index] : parts.field.size,                        filepath,                        extra_upload,                    };                    const result = await ctx.service.changeAtt.save(parts.field, fileData, ctx.session.sessionUser.accountId);                    if (!result) {                        throw '导入数据库保存失败';                    }                    // fileData.in_time = moment(create_time * 1000).format('YYYY-MM-DD');                    // fileData.filesize = await ctx.helper.bytesToSize(fileData.filesize);                    fileData.uid = ctx.session.sessionUser.accountId;                    fileData.id = result.insertId;                    fileData.orginpath = ctx.app.config.fujianOssPath + filepath;                    delete fileData.filepath;                    if (!ctx.helper.canPreview(fileData.fileext)) {                        fileData.filepath = `/change/download/file/${fileData.id}`;                    } else {                        fileData.filepath = ctx.app.config.fujianOssPath + filepath;                    }                    files.push(fileData);                    ++index;                }                responseData.data = files;            } catch (err) {                this.log(err);                // 失败需要消耗掉stream 以防卡死                if (stream) {                    await sendToWormhole(stream);                }                this.setMessage(err.toString(), this.messageType.ERROR);            }            ctx.body = responseData;        }        /**         * 下载附件         * @param {Object} ctx - egg全局变量         * @return {void}         */        async downloadFile(ctx) {            const id = ctx.params.id;            if (id) {                try {                    const fileInfo = await ctx.service.changeAtt.getDataById(id);                    if (fileInfo !== undefined && fileInfo !== '') {                        // const fileName = path.join(this.app.baseDir, fileInfo.filepath);                        // 解决中文无法下载问题                        const userAgent = (ctx.request.header['user-agent'] || '').toLowerCase();                        let disposition = '';                        if (userAgent.indexOf('msie') >= 0 || userAgent.indexOf('chrome') >= 0) {                            disposition = 'attachment; filename=' + encodeURIComponent(fileInfo.filename + fileInfo.fileext);                        } else if (userAgent.indexOf('firefox') >= 0) {                            disposition = 'attachment; filename*="utf8\'\'' + encodeURIComponent(fileInfo.filename + fileInfo.fileext) + '"';                        } else {                            /* safari等其他非主流浏览器只能自求多福了 */                            disposition = 'attachment; filename=' + new Buffer(fileInfo.filename + fileInfo.fileext).toString('binary');                        }                        ctx.response.set({                            'Content-Type': 'application/octet-stream',                            'Content-Disposition': disposition,                            'Content-Length': fileInfo.filesize,                        });                        // ctx.body = await fs.createReadStream(fileName);                        ctx.body = await ctx.helper.ossFileGet(fileInfo.filepath);                    } else {                        throw '不存在该文件';                    }                } catch (err) {                    this.log(err);                    this.setMessage(err.toString(), this.messageType.ERROR);                }            }        }        /**         * 批量下载 - 压缩成zip文件返回         * @param {Oject} ctx - 全局上下文         */        async downloadZip(ctx) {            const time = Date.now();            const zipPath = `app/public/upload/change/fu_jian_zip${time}.zip`;            const responseData = {                err: 0,                msg: '',            };            try {                // const fileIds = JSON.parse(ctx.request.query.fileIds);                const { fileIds = [] } = JSON.parse(ctx.request.body.data);                const { name: changeName } = await ctx.service.changeAtt.getChangeName(ctx.params.cid);                const zipFilename = `${ctx.tender.data.name}-工程变更-${changeName}-附件.zip`;                const size = await ctx.service.changeAtt.compressedFile(fileIds, zipPath);                // 解决中文无法下载问题                const userAgent = (ctx.request.header['user-agent'] || '').toLowerCase();                let disposition = '';                if (userAgent.indexOf('msie') >= 0 || userAgent.indexOf('chrome') >= 0) {                    disposition = 'attachment; filename=' + encodeURIComponent(zipFilename);                } else if (userAgent.indexOf('firefox') >= 0) {                    disposition = 'attachment; filename*="utf8\'\'' + encodeURIComponent(zipFilename) + '"';                } else {                    /* safari等其他非主流浏览器只能自求多福了 */                    disposition = 'attachment; filename=' + new Buffer(zipFilename).toString('binary');                }                ctx.response.set({                    'Content-Type': 'application/octet-stream',                    'Content-Disposition': disposition,                    'Content-Length': size,                });                const readStream = fs.createReadStream(path.join(this.app.baseDir, zipPath));                ctx.body = readStream;                // ctx.body = fs.readFileSync(path.resolve(this.app.baseDir, zipPath));                readStream.on('close', () => {                    if (fs.existsSync(path.resolve(this.app.baseDir, zipPath))) {                        fs.unlinkSync(path.resolve(this.app.baseDir, zipPath));                    }                });                // fs的错误不能被try catch捕捉                readStream.on('error', err => {                    this.log(err);                    if (fs.existsSync(path.resolve(this.app.baseDir, zipPath))) {                        fs.unlinkSync(path.resolve(this.app.baseDir, zipPath));                    }                    responseData.err = 1;                    responseData.msg = err.toString();                    ctx.body = responseData;                });            } catch (err) {                this.log(err);                if (fs.existsSync(path.resolve(this.app.baseDir, zipPath))) {                    fs.unlinkSync(path.resolve(this.app.baseDir, zipPath));                }                this.setMessage(err.toString(), this.messageType.ERROR);                responseData.err = 1;                responseData.msg = err.toString();                ctx.body = responseData;            }        }        /**         * 删除附件         * @param {Object} ctx - egg全局变量         * @return {void}         */        async deleteFile(ctx) {            const responseData = {                err: 0,                msg: '',                data: '',            };            try {                const data = JSON.parse(ctx.request.body.data);                const change = await ctx.service.change.getDataByCondition({ cid: ctx.params.cid });                const fileInfo = await ctx.service.changeAtt.getDataById(data.id);                if (!fileInfo || !Object.keys(fileInfo).length) {                    throw '该文件不存在';                }                if (!fileInfo.extra_upload && change.status === audit.flow.status.checked) {                    throw '无权限删除';                }                if (fileInfo !== undefined && fileInfo !== '') {                    // 先删除文件                    // await fs.unlinkSync(path.join(this.app.baseDir, fileInfo.filepath));                    await ctx.app.fujianOss.delete(ctx.app.config.fujianOssFolder + fileInfo.filepath);                    // 再删除数据库                    await ctx.service.changeAtt.deleteById(data.id);                    responseData.data = '';                } else {                    throw '不存在该文件';                }                // if (data.tid === undefined || data.uci === undefined || data.uc === undefined || data.ac === undefined) {                //     throw '参数有误';                // }                // const [addCompany, selectCompany] = await ctx.service.changeCompany.setCompanyList(data);                // responseData.data = { add: addCompany, select: selectCompany };            } catch (err) {                responseData.err = 1;                responseData.msg = err;            }            ctx.body = responseData;        }        /**         * 查看附件         * @param {Object} ctx - egg全局变量         * @return {void}         */        async checkFile(ctx) {            const responseData = { err: 0, msg: '' };            const id = parseInt(ctx.params.id);            if (id) {                try {                    const fileInfo = await ctx.service.changeAtt.getDataById(id);                    if (fileInfo && Object.keys(fileInfo).length) {                        let filepath = fileInfo.filepath;                        if (!ctx.helper.canPreview(fileInfo.fileext)) {                            filepath = `/change/download/file/${fileInfo.id}`;                        } else {                            filepath = filepath.replace(/^app|\/app/, '');                        }                        fileInfo.filepath && (responseData.data = { filepath });                    }                } catch (error) {                    this.log(error);                    this.setMessage(error.toString(), this.messageType.ERROR);                    responseData.err = 1;                    responseData.msg = error.toString();                }            }            ctx.body = responseData;        }        /**         * 删除变更令         * @param {Object} ctx - egg全局变量         * @return {void}         */        async delete(ctx) {            try {                const result = await ctx.service.change.delete(ctx.request.body.cid);                if (!result) {                    throw '删除变更令失败';                }                ctx.redirect(ctx.request.header.referer);            } catch (err) {                console.log(err);                ctx.redirect(ctx.request.header.referer);            }        }        /**         * 变更令重新审批         * @param {Object} ctx - egg全局变量         * @return {void}         */        async checkAgain(ctx) {            try {                const changeData = await ctx.service.change.getDataByCondition({ cid: ctx.request.body.cid });                if (!changeData) {                    throw '变更令数据错误';                }                // 获取终审                const auditInfo = (await this.ctx.service.changeAudit.getAllDataByCondition({ where: { cid: changeData.cid }, orders: [['usort', 'desc']], limit: 1, offset: 0 }))[0];                if (changeData.status !== audit.flow.status.checked || ctx.session.sessionUser.accountId !== auditInfo.uid) {                    throw '您无权进行该操作';                }                if (ctx.session.sessionUser.loginStatus === 0) {                    const code = ctx.request.body.code;                    const pa = await ctx.service.projectAccount.getDataById(ctx.session.sessionUser.accountId);                    if (!pa.auth_mobile) {                        throw '未绑定手机号';                    }                    const cacheKey = 'smsCode:' + ctx.session.sessionUser.accountId;                    const cacheCode = await app.redis.get(cacheKey);                    // console.log(cacheCode);                    if (cacheCode === null || code === undefined || cacheCode !== (code + pa.auth_mobile)) {                        throw '验证码不正确!';                    }                }                // 获取是否已存在调用变更令                const changeUsedData = await ctx.service.stageChange.getFinalUsedData(ctx.tender.id, changeData.cid);                const stageChangeNum = this.ctx.helper.sum(changeUsedData.map(x => { return Math.abs(x.qty); }));                if (stageChangeNum !== 0) {                    throw '该变更令已被调用,无法重新审批';                }                // 重新审批                const result = await ctx.service.change.checkAgain(changeData.cid);                if (!result) {                    throw '重新审批失败';                }                // ctx.redirect('/tender/' + changeData.tid + '/change/' + changeData.cid + '/info');                ctx.body = {                    err: 0,                    url: ctx.request.header.referer,                    msg: '',                };            } catch (err) {                console.log(err);                // ctx.redirect(ctx.request.header.referer);                ctx.body = {                    err: 1,                    // url: ctx.request.header.referer,                    msg: err,                };            }        }        /**         * 变更令修订重新上报         * @param {Object} ctx - egg全局变量         * @return {void}         */        async checkRevise(ctx) {            try {                const changeData = await ctx.service.change.getDataByCondition({ cid: ctx.request.body.cid });                if (!changeData) {                    throw '变更令数据错误';                }                if (changeData.status !== audit.flow.status.checked || ctx.session.sessionUser.accountId !== changeData.uid) {                    throw '您无权进行该操作';                }                if (ctx.session.sessionUser.loginStatus === 0) {                    const code = ctx.request.body.code;                    const pa = await ctx.service.projectAccount.getDataById(ctx.session.sessionUser.accountId);                    if (!pa.auth_mobile) {                        throw '未绑定手机号';                    }                    const cacheKey = 'smsCode:' + ctx.session.sessionUser.accountId;                    const cacheCode = await app.redis.get(cacheKey);                    // console.log(cacheCode);                    if (cacheCode === null || code === undefined || cacheCode !== (code + pa.auth_mobile)) {                        throw '验证码不正确!';                    }                }                // 获取是否已存在调用变更令                // const changeUsedData = await ctx.service.stageChange.getFinalUsedData(ctx.tender.id, changeData.cid);                // const stageChangeNum = this.ctx.helper.sum(changeUsedData.map(x => { return Math.abs(x.qty); }));                // if (stageChangeNum !== 0) {                //     throw '该变更令已被调用,无法重新审批';                // }                // 重新审批                const result = await ctx.service.change.checkRevise(changeData.cid);                if (!result) {                    throw '修订发起失败';                }                // ctx.redirect('/tender/' + changeData.tid + '/change/' + changeData.cid + '/info');                ctx.body = {                    err: 0,                    url: ctx.request.header.referer,                    msg: '',                };            } catch (err) {                console.log(err);                // ctx.redirect(ctx.request.header.referer);                ctx.body = {                    err: 1,                    // url: ctx.request.header.referer,                    msg: err,                };            }        }        /**         * 获取变更清单         * @param ctx         * @return {Promise<void>}         */        async bills(ctx) {            try {                const data = JSON.parse(ctx.request.body.data);                const responseData = { err: 0, msg: '', data: [] };                switch (data.type) {                    case 'gather':                        responseData.data = await ctx.service.changeAuditList.gatherBgBills(ctx.tender.id);                        break;                    default:                        throw '查询的数据不存在';                }                ctx.body = responseData;            } catch (err) {                this.log(err);                this.ajaxErrorBody(err, '获取变更清单失败');            }        }        /**         * 最后审批人审批成功检查批复编号和其它变更令是否出现重名情况         * @param ctx         * @return {Promise<void>}         */        async checkCodeRepeat(ctx) {            const responseData = {                err: 0,                msg: '',                data: '',            };            try {                const tenderId = ctx.params.id;                const cid = ctx.params.cid;                const data = JSON.parse(ctx.request.body.data);                const result = await ctx.service.change.isRepeat(cid, data.p_code, tenderId);                if (result) {                    throw '该变更令号(批复编号)已使用';                }            } catch (err) {                responseData.err = 1;                responseData.msg = err;            }            ctx.body = responseData;        }        /**         * 拷贝其他变更令         * @param {object} ctx - 全局上下文         */        async copyChange(ctx) {            const responseData = {                err: 0,                msg: '',                data: '',            };            try {                const cid = ctx.params.cid;                const copy_cid = JSON.parse(ctx.request.body.data);                const result = await ctx.service.change.handleCopyChange(cid, copy_cid);                if (!result) {                    responseData.err = 1;                    responseData.msg = '拷贝失败!';                }            } catch (error) {                responseData.err = 1;                responseData.msg = error;            }            ctx.body = responseData;        }        /**         * 修订 详细页面(Get)         * @param ctx         * @return {Promise<void>}         */        async reviseInfo(ctx) {            try {                if (!ctx.session.sessionProject.page_show.openChangeRevise) {                    throw '该功能已关闭';                }                const change = ctx.change;                let edit = true;                let changing = false;                if (change.status !== audit.flow.status.uncheck && change.status !== audit.flow.status.back && change.status !== audit.flow.status.revise) {                    // throw '本条变更审批中或已完成,无法操作台账数据';                    edit = false;                    changing = true;                }                let revising = false;                // 判断是否在修订中,是则无法操作本页                const lastRevise = await ctx.service.ledgerRevise.getLastestRevise(ctx.tender.id);                if (lastRevise && lastRevise.status !== audit.revise.status.checked) {                    // throw '台账修订中,无法操作台账数据';                    edit = false;                    revising = true;                }                const renderData = await this._getDefaultReviseInfoData(ctx, change, edit);                // 台账只读、使用数据                renderData.readOnly = !edit;                renderData.changing = changing;                renderData.revising = revising;                await this.layout('change/revise.ejs', renderData, 'change/revise_modal.ejs');            } catch (err) {                this.log(err);                ctx.redirect(ctx.request.header.referer);            }        }        async _getDefaultReviseInfoData(ctx, change, edit) {            const [ledgerSpread, posSpread] = this._getSpreadSetting(change, edit);            const sjsRela = await this.ctx.service.project.getTenderSjsRela(ctx.session.sessionProject.id);            this.ctx.helper.refreshSpreadShow(sjsRela.ledgerCol, [ledgerSpread, posSpread]);            const [stdBills, stdChapters] = await this.ctx.service.valuation.getValuationStdList(                ctx.tender.data.valuation, ctx.tender.data.measure_type);            return {                change,                tender: ctx.tender.data,                ledgerSpread, posSpread, tenderMenu, measureType,                preUrl: '/tender/' + ctx.tender.id + '/change/' + ctx.change.cid + '/information',                jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.change.revise),                stdBills,                auditConst: audit.flow,                audit: audit.flow,                stdChapters,            };        }        /**         * 获取SpreadSetting         * @private         */        _getSpreadSetting(change, edit) {            const _ = this.app._;            function removeFieldCols(setting, cols) {                _.remove(setting.cols, function(c) {                    return cols.indexOf(c.field) > -1;                });            }            const tender = this.ctx.tender;            const setting = tender.data.measure_type === measureType.tz.value                ? (tender.info.display.ledger.clQty ? spreadConst.withCl : spreadConst.withoutCl)                : (tender.info.display.ledger.clQty ? spreadConst.withClGcl : spreadConst.withoutClGcl);            const ledger = JSON.parse(JSON.stringify(setting.ledger));            const pos = setting.pos ? JSON.parse(JSON.stringify(setting.pos)) : spreadConst.blank;            // if (change.status === audit.flow.status.checking || change.status === audit.flow.status.checked) {            //     ledger.readOnly = true;            //     pos.readOnly = true;            // }            if (!edit) {                ledger.readOnly = true;                pos.readOnly = true;            }            if (tender.data.measure_type === measureType.tz.value) {                removeFieldCols(ledger, spreadConst.filterCols.tzWithoutCols);            }            if (!tender.info.display.ledger.dgnQty) {                removeFieldCols(ledger, spreadConst.filterCols.dgnCols);            }            return [ledger, pos];        }        async updateRevise(ctx) {            try {                if (!ctx.tender.data) throw '标段数据错误';                if (!ctx.session.sessionProject.page_show.openChangeRevise) {                    throw '该功能已关闭';                }                const data = JSON.parse(ctx.request.body.data);                if (!data.postType || !data.postData) throw '数据错误';                const responseData = { err: 0, msg: '', data: {} };                const change = ctx.change;                if (change.status !== audit.flow.status.uncheck && change.status !== audit.flow.status.back && change.status !== audit.flow.status.revise) {                    throw '该变更令正在审批中或已完成,无法操作新增部位数据';                }                // 判断是否在修订中,是则无法操作本页                const lastRevise = await ctx.service.ledgerRevise.getLastestRevise(ctx.tender.id);                if (lastRevise && lastRevise.status !== audit.revise.status.checked) {                    throw '台账修订中,无法操作新增部位数据';                }                // const revise = await this.checkRevise(ctx);                //                switch (data.postType) {                    case 'add':                    case 'delete':                    case 'up-move':                    case 'down-move':                    case 'up-level':                    case 'down-level':                        responseData.data = await this._billsBase(ctx, data.postType, data.postData);                        break;                    case 'update':                        ctx.helper.checkDgnQtyPrecision(data.postData);                        responseData.data = await ctx.service.changeLedger.updateCalc(ctx.tender.id, data.postData);                        break;                    case 'paste-block':                        responseData.data = await this._pasteBlock(ctx, data.postData);                        break;                    case 'add-std':                        responseData.data = await this._addStd(ctx, data.postData);                        break;                    case 'add-deal':                        responseData.data = await this._addDeal(ctx, data.postData);                        break;                    // case 'batch-insert':                    //     responseData.data = await this._batchInsert(ctx, data.postData);                    //     break;                    case 'pos':                        responseData.data = await this._updatePos(ctx, data);                        break;                    default:                        throw '未知操作';                }                ctx.body = responseData;            } catch (err) {                this.log(err);                ctx.body = this.ajaxErrorBody(err, '数据错误');            }        }        async _billsBase(ctx, type, data) {            if (isNaN(data.id) || data.id <= 0) throw '数据错误';            if (type !== 'add') {                if (isNaN(data.count) || data.count <= 0) data.count = 1;            }            switch (type) {                case 'add':                    return await ctx.service.changeLedger.addNodeBatch(ctx.tender.id, data.id, { ccid: ctx.change.cid, check_calc: 1 }, data.count);                case 'delete':                    return await ctx.service.changeLedger.delete(ctx.tender.id, data.id, data.count);                case 'up-move':                    return await ctx.service.changeLedger.upMoveNode(ctx.tender.id, data.id, data.count);                case 'down-move':                    return await ctx.service.changeLedger.downMoveNode(ctx.tender.id, data.id, data.count);                case 'up-level':                    return await ctx.service.changeLedger.upLevelNode(ctx.tender.id, data.id, data.count);                case 'down-level':                    return await ctx.service.changeLedger.downLevelNode(ctx.tender.id, data.id, data.count);                default:                    throw '未知操作';            }        }        /**         * 复制粘贴整块         *         * @param ctx         * @return {Promise<void>}         */        async _pasteBlock(ctx, data) {            if ((isNaN(data.id) || data.id <= 0) ||                (!data.tid && data.tid <= 0) ||                (!data.block || data.block.length <= 0)) throw '参数错误';            return await ctx.service.changeLedger.pasteBlockData(ctx.tender.id, data.id, data.block);        }        /**         * 从标准项目表添加数据         * @param ctx         * @return {Promise<void>}         */        async _addStd(ctx, data) {            if ((isNaN(data.id) || data.id <= 0) || !data.stdType || !data.stdNode) throw '参数错误';            // todo 校验项目是否使用该库的权限            let stdLib,                addType;            switch (data.stdType) {                case 'xmj':                    stdLib = ctx.service.stdXmj;                    addType = stdDataAddType.withParent;                    break;                case 'gcl':                    stdLib = ctx.service.stdGcl;                    const selectNode = await ctx.service.changeLedger.getDataByNodeId(ctx.tender.id, data.id) || await ctx.service.ledger.getDataByNodeId(ctx.tender.id, data.id);                    if (selectNode.b_code) {                        addType = stdDataAddType.next;                    } else {                        addType = stdDataAddType.child;                    }                    break;                default:                    throw '未知标准库';            }            const stdData = await stdLib.getDataByDataId(data.stdLibId, data.stdNode);            switch (addType) {                case stdDataAddType.child:                    return await ctx.service.changeLedger.addStdNodeAsChild(ctx.tender.id, data.id, stdData, ctx.change.cid);                case stdDataAddType.next:                    return await ctx.service.changeLedger.addStdNode(ctx.tender.id, data.id, stdData, ctx.change.cid);                case stdDataAddType.withParent:                    return await ctx.service.changeLedger.addStdNodeWithParent(ctx.tender.id, stdData, stdLib, ctx.change.cid);                default:                    throw '未知添加方式';            }        }        /**         * 从签约清单添加节点         * @param ctx         * @return {Promise<void>}         */        async _addDeal(ctx, data) {            if (!data.type || !data.dealBills) throw '数据错误';            data.dealBills.unit_price = this.ctx.helper.round(data.dealBills.unit_price, ctx.tender.info.decimal.up);            if (data.type === 'child') {                return await ctx.service.changeLedger.addChild(ctx.tender.id, data.id, data.dealBills, ctx.change.cid);            } else if (data.type === 'next') {                data.dealBills.ccid = ctx.change.cid;                return await ctx.service.changeLedger.addNode(ctx.tender.id, data.id, data.dealBills);            }            throw '数据错误';        }        /**         * 批量插入数据         *         * data = {id, batchData, batchType}         * data.batchType = 'batchInsertChild'/'batchInsertNext'         * data.batchData = [{name, children}] -- 项目节列表         * data.batchData.children = [{code, name, unit, unit_price, quantity}] -- 工程量清单列表         *         * @param ctx         * @return {Promise<void>}         */        async _batchInsert(ctx, data) {            if ((isNaN(data.id) || data.id <= 0) || !data.batchType) throw '参数错误';            switch (data.batchType) {                case 'child':                    return await ctx.service.changeLedger.batchInsertChild(ctx.tender.id, data.id, data.batchData);                case 'next':                    return await ctx.service.changeLedger.batchInsertNext(ctx.tender.id, data.id, data.batchData);                default:                    throw '参数错误';            }        }        async _updatePos(ctx, data) {            await this.checkMeasureType(measureType.tz.value);            if (!data.posPostType) throw '数据错误';            switch (data.posPostType) {                case 'add':                    return await this.ctx.service.changePos.addPos(ctx.tender.id, ctx.change.cid, data.postData);                case 'update':                    if (data.postData instanceof Array) {                        return await this.ctx.service.changePos.updatePosArr(ctx.tender.id, data.postData);                    }                    return await this.ctx.service.changePos.updatePos(ctx.tender.id, data.postData);                case 'delete':                    return await this.ctx.service.changePos.deletePos(ctx.tender.id, data.postData);                case 'paste':                    return await this.ctx.service.changePos.pastePosData(ctx.tender.id, ctx.change.cid, data.postData);                default:                    throw '未知操作';            }        }        async _filterChangesProject(ctx, status = 0) {            const tenderId = ctx.params.id;            ctx.session.sessionUser.tenderId = tenderId;            const tender = await this.service.tender.getDataById(ctx.tender.id);            // const tender = ctx.tender;            // const tenderList = await this.service.tender.getList();            const page = ctx.page;            const sorts = ctx.query.sort ? ctx.query.sort : 0;            const orders = ctx.query.order ? ctx.query.order : 0;            const changes = await ctx.service.changeProject.getListByStatus(tender.id, status, 1, sorts, orders);            const total = await ctx.service.changeProject.getCountByStatus(tender.id, status);            for (const c of changes) {                c.curAuditor = await ctx.service.changeProjectAudit.getAuditorByStatus(c.id, c.status, c.times);            }            const accountInfo = await this.ctx.service.projectAccount.getDataById(this.ctx.session.sessionUser.accountId);            const userPermission = accountInfo !== undefined && accountInfo.permission !== ''                ? JSON.parse(accountInfo.permission) : null;            // 分页相关            const pageInfo = {                page,                total: Math.ceil(total / app.config.pageSize),                queryData: JSON.stringify(ctx.urlInfo.query),            };            const filter = JSON.parse(JSON.stringify(audit.changeProject.filter));            filter.count = [];            filter.count[filter.status.pending] = await ctx.service.changeProject.getCountByStatus(tender.id, filter.status.pending);// await ctx.service.change.pendingDatas(tender.id, ctx.session.sessionUser.accountId);            filter.count[filter.status.uncheck] = await ctx.service.changeProject.getCountByStatus(tender.id, filter.status.uncheck);// await ctx.service.change.checkingDatas(tender.id, ctx.session.sessionUser.accountId);            filter.count[filter.status.checking] = await ctx.service.changeProject.getCountByStatus(tender.id, filter.status.checking);// await ctx.service.change.checkedDatas(tender.id, ctx.session.sessionUser.accountId);            filter.count[filter.status.checked] = await ctx.service.changeProject.getCountByStatus(tender.id, filter.status.checked);// await ctx.service.change.pendingDatas(tender.id, ctx.session.sessionUser.accountId);            filter.count[filter.status.checkNo] = await ctx.service.changeProject.getCountByStatus(tender.id, filter.status.checkNo);// await ctx.service.change.pendingDatas(tender.id, ctx.session.sessionUser.accountId);            let codeRule = [];            let c_connector = '1';            let c_rule_first = 1;            const rule_type = tender.user_id === ctx.session.sessionUser.accountId ? 'suggestion' : 'will';            if (tender.c_code_rules) {                const c_code_rules = JSON.parse(tender.c_code_rules);                codeRule = c_code_rules[rule_type + '_rule'] !== undefined ? c_code_rules[rule_type + '_rule'] : [];                c_connector = c_code_rules[rule_type + '_connector'] !== undefined ? c_code_rules[rule_type + '_connector'] : '1';                c_rule_first = c_code_rules[rule_type + '_rule_first'] !== undefined ? c_code_rules[rule_type + '_rule_first'] : 1;            }            for (const rule of codeRule) {                switch (rule.rule_type) {                    case codeRuleConst.measure.ruleType.dealCode:                        rule.preview = ctx.tender.info.deal_info.dealCode;                        break;                    case codeRuleConst.measure.ruleType.tenderName:                        rule.preview = tender.name;                        break;                    case codeRuleConst.measure.ruleType.inDate:                        rule.preview = moment().format('YYYY');                        break;                    case codeRuleConst.measure.ruleType.text:                        rule.preview = rule.text;                        break;                    case codeRuleConst.measure.ruleType.addNo:                        const s = '0000000000';                        rule.preview = s.substr(s.length - rule.format);                        break;                    default: break;                }            }            const renderData = {                uid: ctx.session.sessionUser.accountId,                userPermission,                tender,                pageInfo,                changes,                status,                rule_type,                codeRule,                c_connector,                c_rule_first,                filter,                ruleType: codeRuleConst.ruleType[rule_type],                dealCode: ctx.tender.info.deal_info.dealCode,                auditConst: audit.changeProject,                ruleConst: codeRuleConst.measure,                changeConst,                tenderMenu: this.menu.tenderMenu,                preUrl: '/tender/' + tender.id,                jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.change.project),            };            await this.layout('change/project.ejs', renderData, 'change/project_modal.ejs');        }        /**         * 变更立项列表 页面 (Get)         *         * @param {Object} ctx - egg全局变量         * @return {void}         */        async project(ctx) {            try {                await this._filterChangesProject(ctx);            } catch (err) {                this.log(err);                ctx.redirect('/dashboard');            }        }        /**         * 新增变更立项 (Post)         *         * @param {Object} ctx - egg全局变量         * @return {void}         */        async projectAdd(ctx) {            try {                const tenderId = ctx.params.id;                if (!tenderId) {                    throw '当前未打开标段';                }                const data = JSON.parse(ctx.request.body.data);                if (!data.code || data.code === '' || !data.name || data.name === '') {                    throw '变更立项书编号不能为空';                }                const change = await ctx.service.changeProject.add(tenderId, ctx.session.sessionUser.accountId, data.code, data.name);                ctx.body = { err: 0, msg: '', data: change };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString() };            }        }        /**         * 删除变更立项         * @param {Object} ctx - egg全局变量         * @return {void}         */        async projectDelete(ctx) {            try {                const result = await ctx.service.changeProject.delete(ctx.request.body.cpid);                if (!result) {                    throw '删除变更立项失败';                }                ctx.redirect(ctx.request.header.referer);            } catch (err) {                console.log(err);                ctx.redirect(ctx.request.header.referer);            }        }        /**         * 变更管理 状态筛选 页面 (Get)         * @param {Object} ctx - egg全局变量         * @return {void}         */        async projectStatus(ctx) {            try {                const status = parseInt(ctx.params.status);                await this._filterChangesProject(ctx, status);            } catch (err) {                this.logger.error(err);                ctx.redirect('/tender/' + ctx.params.id + '/change/project');            }        }        /**         * 获取审批界面所需的 原报、审批人数据等         * @param ctx         * @return {Promise<void>}         * @private         */        async _getChangeProjectAuditViewData(ctx) {            const auditConst = audit.changeProject;            const times = ctx.change.status === auditConst.status.back ? ctx.change.times - 1 : ctx.change.times;            ctx.change.user = await ctx.service.projectAccount.getAccountInfoById(ctx.change.uid);            ctx.change.auditHistory = [];            if (times >= 1) {                for (let i = 1; i <= times; i++) {                    ctx.change.auditHistory.push(await ctx.service.changeProjectAudit.getAuditors(ctx.change.id, i));                }            }            // 获取审批流程中左边列表            ctx.change.auditors2 = ctx.change.status === auditConst.status.back && ctx.change.uid !== ctx.session.sessionUser.accountId ?                await ctx.service.changeProjectAudit.getAuditorsWithOwner(ctx.change.id, times) :                await ctx.service.changeProjectAudit.getAuditorsWithOwner(ctx.change.id, ctx.change.times);            if (ctx.change.status === auditConst.status.uncheck || ctx.change.status === auditConst.status.back) {                ctx.change.auditorList = await ctx.service.changeProjectAudit.getAuditors(ctx.change.id, ctx.change.times);            }            ctx.change.xsAuditors = await ctx.service.changeProjectXsAudit.getAuditList(ctx.change.id);        }        async projectInformation(ctx) {            try {                const whiteList = this.ctx.app.config.multipart.whitelist;                const tender = await ctx.service.tender.getDataById(ctx.tender.id);                // 获取附件列表                const fileList = await ctx.service.changeProjectAtt.getAllChangeProjectAtt(ctx.tender.id, ctx.change.id);                await this._getChangeProjectAuditViewData(ctx);                const renderData = {                    tender,                    change: ctx.change,                    changeConst,                    auditConst: audit.changeProject,                    fileList,                    whiteList,                    returnUrl: this.app._.includes(ctx.request.headers.referer, '/tender/' + ctx.tender.id + '/change/project') ? ctx.request.headers.referer : null,                    jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.change.project_information),                    preUrl: '/tender/' + ctx.tender.id + '/change/project/' + ctx.change.id + '/information',                };                // if ((ctx.change.status === audit.changeProject.status.uncheck || ctx.change.status === audit.changeProject.status.back) && (ctx.session.sessionUser.accountId === ctx.change.uid || ctx.tender.isTourist)) {                    // data.accountGroup = accountGroup;                    // 获取所有项目参与者                const accountList = await ctx.service.projectAccount.getAllDataByCondition({                    where: { project_id: ctx.session.sessionProject.id, enable: 1 },                    columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group', 'mobile'],                });                renderData.accountList = accountList;                renderData.accountGroup = accountGroup.map((item, idx) => {                    const groupList = accountList.filter(item => item.account_group === idx);                    return { groupName: item, groupList };                });                // }                await this.layout('change/project_information.ejs', renderData, 'change/project_information_modal.ejs');            } catch (err) {                this.log(err);                ctx.redirect('/tender/' + ctx.params.id + '/change');            }        }        // 审批相关        /**         * 添加审批人         * @param ctx         * @return {Promise<void>}         */        async addProjectAudit(ctx) {            try {                const auditConst = audit.changeProject;                const data = JSON.parse(ctx.request.body.data);                const id = this.app._.toInteger(data.auditorId);                if (isNaN(id) || id <= 0) {                    throw '参数错误';                }                // 检查权限等                if (ctx.change.uid !== ctx.session.sessionUser.accountId) {                    throw '您无权添加审核人';                }                if (ctx.change.status === auditConst.status.checking || ctx.change.status === auditConst.status.checked) {                    throw '当前不允许添加审核人';                }                // 判断是否是协审人,是则无法添加到审批人中                const xsAuditorList = await ctx.service.changeProjectXsAudit.getAuditList(ctx.change.id);                const xsAidList = this.app._.map(xsAuditorList, 'aid');                if (this.app._.indexOf(xsAidList, id) !== -1) {                    throw '该用户已添加到协审人列表中,请删除该协审人再添加';                }                ctx.change.auditorList = await ctx.service.changeProjectAudit.getAuditors(ctx.change.id, ctx.change.times);                // 检查审核人是否已存在                const exist = this.app._.find(ctx.change.auditorList, { aid: id });                if (exist) {                    throw '该审核人已存在,请勿重复添加';                }                // const shenpiInfo = await ctx.service.shenpiAudit.getDataByCondition({ tid: ctx.tender.id, sp_type: shenpiConst.sp_type.material, sp_status: shenpiConst.sp_status.gdzs });                // const is_gdzs = shenpiInfo && ctx.tender.info.shenpi.material === shenpiConst.sp_status.gdzs ? 1 : 0;                const result = await ctx.service.changeProjectAudit.addAuditor(ctx.change.id, id, ctx.change.times);                if (!result) {                    throw '添加审核人失败';                }                const auditors = await ctx.service.changeProjectAudit.getAuditorsWithOwner(ctx.change.id, ctx.change.times);                ctx.body = { err: 0, msg: '', data: auditors };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 移除审批人         * @param ctx         * @return {Promise<void>}         */        async deleteProjectAudit(ctx) {            try {                const data = JSON.parse(ctx.request.body.data);                const id = data.auditorId instanceof Number ? data.auditorId : this.app._.toNumber(data.auditorId);                if (isNaN(id) || id <= 0) {                    throw '参数错误';                }                const result = await ctx.service.changeProjectAudit.deleteAuditor(ctx.change.id, id, ctx.change.times);                if (!result) {                    throw '移除审核人失败';                }                const auditors = await ctx.service.changeProjectAudit.getAuditors(ctx.change.id, ctx.change.times);                ctx.body = { err: 0, msg: '', data: auditors };            } catch (err) {                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 添加协审人         * @param ctx         * @return {Promise<void>}         */        async addProjectXsAudit(ctx) {            try {                const auditConst = audit.changeProject;                const data = JSON.parse(ctx.request.body.data);                const id = this.app._.toInteger(data.auditorId);                if (isNaN(id) || id <= 0) {                    throw '参数错误';                }                const times = ctx.change.status === auditConst.status.back ? ctx.change.times - 1 : ctx.change.times;                const spAuditors = await ctx.service.changeProjectAudit.getAuditorsWithOwner(ctx.change.id, times);                const aidLists = this.app._.map(spAuditors, 'aid');                // 检查权限等                if (this.app._.indexOf(aidLists, ctx.session.sessionUser.accountId) === -1) {                    throw '您无权添加协审人';                }                // 如果已经是审批人则无法添加此人为协审                if (this.app._.indexOf(aidLists, id) !== -1) {                    throw '该用户为审批人或原报,无需添加至协审人';                }                const auditorList = await ctx.service.changeProjectXsAudit.getAuditList(ctx.change.id);                // 检查审核人是否已存在                const exist = this.app._.find(auditorList, { aid: id });                if (exist) {                    throw '该协审人已存在,请勿重复添加';                }                const result = await ctx.service.changeProjectXsAudit.addAuditor(ctx.change.id, id);                if (!result) {                    throw '添加协审人失败';                }                const auditors = await ctx.service.changeProjectXsAudit.getOneAudit(ctx.change.id, id);                ctx.body = { err: 0, msg: '', data: auditors };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 移除协审人         * @param ctx         * @return {Promise<void>}         */        async deleteProjectXsAudit(ctx) {            try {                const data = JSON.parse(ctx.request.body.data);                const id = data.auditorId instanceof Number ? data.auditorId : this.app._.toNumber(data.auditorId);                if (isNaN(id) || id <= 0) {                    throw '参数错误';                }                const result = await ctx.service.changeProjectXsAudit.deleteAuditor(ctx.change.id, id);                if (!result) {                    throw '移除协审人失败';                }                const auditors = await ctx.service.changeProjectXsAudit.getAuditList(ctx.change.id);                ctx.body = { err: 0, msg: '', data: auditors };            } catch (err) {                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 上传附件         * @param {*} ctx 上下文         */        async uploadProjectFile(ctx) {            let stream;            try {                const auditConst = audit.changeProject;                // this._checkAdvanceFileCanModify(ctx);                const parts = this.ctx.multipart({                    autoFields: true,                });                const files = [];                const create_time = Date.parse(new Date()) / 1000;                let idx = 0;                const extra_upload = ctx.change.status === auditConst.status.checked;                while ((stream = await parts()) !== undefined) {                    if (!stream.filename) {                        // 如果没有传入直接返回                        return;                    }                    const fileInfo = path.parse(stream.filename);                    const filepath = `app/public/upload/${this.ctx.tender.id.toString()}/change_project/fujian_${create_time + idx.toString() + fileInfo.ext}`;                    // await ctx.helper.saveStreamFile(stream, path.resolve(this.app.baseDir, 'app', filepath));                    await ctx.app.fujianOss.put(ctx.app.config.fujianOssFolder + filepath, stream);                    files.push({ filepath, name: stream.filename, ext: fileInfo.ext });                    ++idx;                    stream && (await sendToWormhole(stream));                }                const in_time = new Date();                const payload = files.map(file => {                    let idx;                    if (Array.isArray(parts.field.name)) {                        idx = parts.field.name.findIndex(name => name === file.name);                    } else {                        idx = 'isString';                    }                    const newFile = {                        tid: ctx.tender.id,                        cpid: ctx.change.id,                        uid: ctx.session.sessionUser.accountId,                        filename: file.name,                        fileext: file.ext,                        filesize: ctx.helper.bytesToSize(idx === 'isString' ? parts.field.size : parts.field.size[idx]),                        filepath: file.filepath,                        upload_time: in_time,                        extra_upload,                    };                    return newFile;                });                // 执行文件信息写入数据库                await ctx.service.changeProjectAtt.saveFileMsgToDb(payload);                // 将最新的当前标段的所有文件信息返回                const data = await ctx.service.changeProjectAtt.getAllChangeProjectAtt(ctx.tender.id, ctx.change.id);                ctx.body = { err: 0, msg: '', data };            } catch (err) {                stream && (await sendToWormhole(stream));                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 删除附件         * @param {Ojbect} ctx 上下文         */        async deleteProjectFile(ctx) {            try {                const { id } = JSON.parse(ctx.request.body.data);                const fileInfo = await ctx.service.changeProjectAtt.getDataById(id);                if (fileInfo || Object.keys(fileInfo).length) {                    // 先删除文件                    // await fs.unlinkSync(path.resolve(this.app.baseDir, './app', fileInfo.filepath));                    await ctx.app.fujianOss.delete(ctx.app.config.fujianOssFolder + fileInfo.filepath);                    // 再删除数据库                    await ctx.service.changeProjectAtt.delete(id);                } else {                    throw '不存在该文件';                }                const data = await ctx.service.changeProjectAtt.getAllChangeProjectAtt(ctx.tender.id, ctx.change.id);                ctx.body = { err: 0, msg: '请求成功', data };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 下载附件         * @param {Object} ctx - egg全局变量         * @return {void}         */        async downloadProjectFile(ctx) {            const id = ctx.params.fid;            if (id) {                try {                    const fileInfo = await ctx.service.changeProjectAtt.getDataById(id);                    if (fileInfo !== undefined && fileInfo !== '') {                        // const fileName = path.join(__dirname, '../', fileInfo.filepath);                        // 解决中文无法下载问题                        const userAgent = (ctx.request.header['user-agent'] || '').toLowerCase();                        let disposition = '';                        if (userAgent.indexOf('msie') >= 0 || userAgent.indexOf('chrome') >= 0) {                            disposition = 'attachment; filename=' + encodeURIComponent(fileInfo.filename);                        } else if (userAgent.indexOf('firefox') >= 0) {                            disposition = 'attachment; filename*="utf8\'\'' + encodeURIComponent(fileInfo.filename) + '"';                        } else {                            /* safari等其他非主流浏览器只能自求多福了 */                            disposition = 'attachment; filename=' + new Buffer(fileInfo.filename).toString('binary');                        }                        ctx.response.set({                            'Content-Type': 'application/octet-stream',                            'Content-Disposition': disposition,                            'Content-Length': fileInfo.filesize,                        });                        // ctx.body = await fs.createReadStream(fileName);                        ctx.body = await ctx.helper.ossFileGet(fileInfo.filepath);                    } else {                        throw '不存在该文件';                    }                } catch (err) {                    this.log(err);                    this.setMessage(err.toString(), this.messageType.ERROR);                }            }        }        async projectInformationSave(ctx) {            try {                const data = JSON.parse(ctx.request.body.data);                if (data.name === 'code') {                    const info = await ctx.service.changeProject.isRepeat(ctx.change.id, data.val, ctx.tender.id, ctx.change.type);                    if (info) {                        throw '该编号已存在';                    }                }                const result = await ctx.service.changeProject.saveInfo(ctx.change.id, data);                if (!result) {                    throw '修改失败';                }                ctx.body = { err: 0, msg: '请求成功', data: null };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 上报和重新上报         * @param ctx         * @return {Promise<void>}         */        async startProjectAudit(ctx) {            try {                const auditConst = audit.changeProject;                // 检查权限等                if (!ctx.change) {                    throw '数据错误';                }                if (ctx.change.uid !== ctx.session.sessionUser.accountId) {                    throw '您无权上报该期数据';                }                if (ctx.change.status === auditConst.status.checking || ctx.change.status === auditConst.status.checked) {                    throw '该材料调差期数据当前无法上报';                }                await ctx.service.changeProjectAudit.start(ctx.change.id, ctx.change.times);                ctx.redirect(ctx.request.header.referer);            } catch (err) {                this.log(err);                ctx.session.postError = err.toString();                ctx.redirect(ctx.request.header.referer);            }        }        /**         * 审批         * @param ctx         * @return {Promise<void>}         */        async checkProjectAudit(ctx) {            try {                const auditConst = audit.changeProject;                if (!ctx.change || ctx.change.status !== auditConst.status.checking) {                    throw '当前材料调差期数据有误';                }                if (!ctx.change.curAuditor || ctx.change.curAuditor.aid !== ctx.session.sessionUser.accountId) {                    throw '您无权进行该操作';                }                const data = {                    checkType: parseInt(ctx.request.body.checkType),                    opinion: ctx.request.body.opinion,                };                if (!data.checkType || isNaN(data.checkType)) {                    throw '提交数据错误';                }                // if (data.checkType === auditConst.status.checkNo) {                //     if (!data.checkType || isNaN(data.checkType)) {                //         throw '提交数据错误';                //     }                // }                await ctx.service.changeProjectAudit.check(ctx.change.id, data, ctx.change.times);                ctx.redirect(ctx.request.header.referer);            } catch (err) {                this.log(err);                ctx.session.postError = err.toString();                ctx.redirect(ctx.request.header.referer);            }        }        async _filterChangesApply(ctx, status = 0) {            const tenderId = ctx.params.id;            ctx.session.sessionUser.tenderId = tenderId;            const tender = await this.service.tender.getDataById(ctx.tender.id);            // const tender = ctx.tender;            // const tenderList = await this.service.tender.getList();            const page = ctx.page;            const sorts = ctx.query.sort ? ctx.query.sort : 0;            const orders = ctx.query.order ? ctx.query.order : 0;            const changes = await ctx.service.changeApply.getListByStatus(tender.id, status, 1, sorts, orders);            const total = await ctx.service.changeApply.getCountByStatus(tender.id, status);            let page_total = 0;            const tp = await ctx.service.changeApply.getTp(tender.id, status);            for (const c of changes) {                c.curAuditor = await ctx.service.changeApplyAudit.getAuditorByStatus(c.id, c.status, c.times);                page_total = ctx.helper.add(page_total, c.total_price);            }            const tender_userInfo = await ctx.service.projectAccount.getDataById(ctx.tender.data.user_id);            // 分页相关            const pageInfo = {                page,                total: Math.ceil(total / app.config.pageSize),                queryData: JSON.stringify(ctx.urlInfo.query),            };            const filter = JSON.parse(JSON.stringify(audit.changeApply.filter));            filter.count = [];            filter.count[filter.status.pending] = await ctx.service.changeApply.getCountByStatus(tender.id, filter.status.pending);// await ctx.service.change.pendingDatas(tender.id, ctx.session.sessionUser.accountId);            filter.count[filter.status.uncheck] = await ctx.service.changeApply.getCountByStatus(tender.id, filter.status.uncheck);// await ctx.service.change.checkingDatas(tender.id, ctx.session.sessionUser.accountId);            filter.count[filter.status.checking] = await ctx.service.changeApply.getCountByStatus(tender.id, filter.status.checking);// await ctx.service.change.checkedDatas(tender.id, ctx.session.sessionUser.accountId);            filter.count[filter.status.checked] = await ctx.service.changeApply.getCountByStatus(tender.id, filter.status.checked);// await ctx.service.change.pendingDatas(tender.id, ctx.session.sessionUser.accountId);            // filter.count[filter.status.checkNo] = await ctx.service.changeApply.getCountByStatus(tender.id, filter.status.checkNo);// await ctx.service.change.pendingDatas(tender.id, ctx.session.sessionUser.accountId);            let codeRule = [];            let c_connector = '1';            let c_rule_first = 1;            const rule_type = 'apply';            if (tender.c_code_rules) {                const c_code_rules = JSON.parse(tender.c_code_rules);                codeRule = c_code_rules[rule_type + '_rule'] !== undefined ? c_code_rules[rule_type + '_rule'] : [];                c_connector = c_code_rules[rule_type + '_connector'] !== undefined ? c_code_rules[rule_type + '_connector'] : '1';                c_rule_first = c_code_rules[rule_type + '_rule_first'] !== undefined ? c_code_rules[rule_type + '_rule_first'] : 1;            }            for (const rule of codeRule) {                switch (rule.rule_type) {                    case codeRuleConst.measure.ruleType.dealCode:                        rule.preview = ctx.tender.info.deal_info.dealCode;                        break;                    case codeRuleConst.measure.ruleType.tenderName:                        rule.preview = tender.name;                        break;                    case codeRuleConst.measure.ruleType.inDate:                        rule.preview = moment().format('YYYY');                        break;                    case codeRuleConst.measure.ruleType.text:                        rule.preview = rule.text;                        break;                    case codeRuleConst.measure.ruleType.addNo:                        const s = '0000000000';                        rule.preview = s.substr(s.length - rule.format);                        break;                    default: break;                }            }            const changeProjectList = await ctx.service.changeProject.getAllDataByCondition({ where: { tid: tender.id, status: audit.changeProject.status.checked } });            const allProjectCodes = await ctx.service.changeApply.getAllDataByCondition({                columns: ['project_code'],                where: {                    tid: tender.id,                },            });            const pcLists = allProjectCodes.length > 0 ? ctx.app._.uniq(ctx.app._.map(allProjectCodes, 'project_code')) : [];            const renderData = {                uid: ctx.session.sessionUser.accountId,                tender,                pageInfo,                changes,                status,                rule_type,                codeRule,                c_connector,                c_rule_first,                filter,                apply_username: tender_userInfo ? tender_userInfo.name : null,                ruleType: codeRuleConst.ruleType[rule_type],                dealCode: ctx.tender.info.deal_info.dealCode,                auditConst: audit.changeApply,                ruleConst: codeRuleConst.measure,                changeConst,                changeProjectList,                pcLists,                tp,                page_total,                tenderMenu: this.menu.tenderMenu,                preUrl: '/tender/' + tender.id,                jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.change.apply),            };            await this.layout('change/apply.ejs', renderData, 'change/apply_modal.ejs');        }        /**         * 变更申请列表 页面 (Get)         *         * @param {Object} ctx - egg全局变量         * @return {void}         */        async apply(ctx) {            try {                await this._filterChangesApply(ctx);            } catch (err) {                this.log(err);                ctx.redirect('/dashboard');            }        }        /**         * 变更管理 状态筛选 页面 (Get)         * @param {Object} ctx - egg全局变量         * @return {void}         */        async applyStatus(ctx) {            try {                const status = parseInt(ctx.params.status);                await this._filterChangesApply(ctx, status);            } catch (err) {                this.logger.error(err);                ctx.redirect('/tender/' + ctx.params.id + '/change/apply');            }        }        /**         * 新增变更立项 (Post)         *         * @param {Object} ctx - egg全局变量         * @return {void}         */        async applyAdd(ctx) {            try {                const tenderId = ctx.params.id;                if (!tenderId) {                    throw '当前未打开标段';                }                const data = JSON.parse(ctx.request.body.data);                if (!data.code || data.code === '') {                    throw '变更申请编号不能为空';                }                const change = await ctx.service.changeApply.add(tenderId, ctx.session.sessionUser.accountId, data.code, data.project_code, data.name);                ctx.body = { err: 0, msg: '', data: change };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString() };            }        }        /**         * 删除变更立项         * @param {Object} ctx - egg全局变量         * @return {void}         */        async applyDelete(ctx) {            try {                const result = await ctx.service.changeApply.delete(ctx.request.body.caid);                if (!result) {                    throw '删除变更立项失败';                }                ctx.redirect(ctx.request.header.referer);            } catch (err) {                console.log(err);                ctx.redirect(ctx.request.header.referer);            }        }        /**         * 获取审批界面所需的 原报、审批人数据等         * @param ctx         * @return {Promise<void>}         * @private         */        async _getChangeApplyAuditViewData(ctx) {            const auditConst = audit.changeApply;            const times = ctx.change.status === auditConst.status.checkNo ? ctx.change.times - 1 : ctx.change.times;            ctx.change.user = await ctx.service.projectAccount.getAccountInfoById(ctx.change.uid);            ctx.change.auditHistory = [];            if (times >= 1) {                for (let i = 1; i <= times; i++) {                    ctx.change.auditHistory.push(await ctx.service.changeApplyAudit.getAuditors(ctx.change.id, i));                }            }            // 获取审批流程中左边列表            ctx.change.auditors2 = ctx.change.status === auditConst.status.checkNo && ctx.change.uid !== ctx.session.sessionUser.accountId ?                await ctx.service.changeApplyAudit.getAuditorsWithOwner(ctx.change.id, times) :                await ctx.service.changeApplyAudit.getAuditorsWithOwner(ctx.change.id, ctx.change.times);            if (ctx.change.status === auditConst.status.uncheck || ctx.change.status === auditConst.status.checkNo) {                ctx.change.auditorList = await ctx.service.changeApplyAudit.getAuditors(ctx.change.id, ctx.change.times);            }        }        async applyInformation(ctx) {            try {                const whiteList = this.ctx.app.config.multipart.whitelist;                const tender = await ctx.service.tender.getDataById(ctx.tender.id);                await this._getChangeApplyAuditViewData(ctx);                // 获取清单列表                const changeList = await ctx.service.changeApplyList.getList(ctx.change.id);                // 获取附件列表                const fileList = await ctx.service.changeApplyAtt.getAllChangeApplyAtt(ctx.tender.id, ctx.change.id);                const renderData = {                    tender,                    change: ctx.change,                    listRule: tender.c_apply_list_rule ? JSON.parse(tender.c_apply_list_rule) : { source: 1, rule: ['unit', 'unit_price'] },                    changeList,                    changeConst,                    auditConst: audit.changeApply,                    fileList,                    whiteList,                    tpUnit: ctx.change.decimal ? ctx.change.decimal.tp : ctx.tender.info.decimal.tp,                    upUnit: ctx.change.decimal ? ctx.change.decimal.up : ctx.tender.info.decimal.up,                    changeUnits: changeConst.units,                    precision: ctx.change.decimal && ctx.change.decimal.precision ? ctx.change.decimal.precision : ctx.tender.info.precision,                    returnUrl: this.app._.includes(ctx.request.headers.referer, '/tender/' + ctx.tender.id + '/change/apply') && !this.app._.includes(ctx.request.headers.referer, '/tender/' + ctx.tender.id + '/change/apply/' + ctx.change.id + '/information') ? ctx.request.headers.referer : null,                    jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.change.apply_information),                    preUrl: '/tender/' + ctx.tender.id + '/change/apply/' + ctx.change.id + '/information',                };                if ((ctx.change.status === audit.changeApply.status.uncheck || ctx.change.status === audit.changeApply.status.checkNo) && (ctx.session.sessionUser.accountId === ctx.change.uid || ctx.tender.isTourist)) {                    // data.accountGroup = accountGroup;                    // 获取所有项目参与者                    const accountList = await ctx.service.projectAccount.getAllDataByCondition({                        where: { project_id: ctx.session.sessionProject.id, enable: 1 },                        columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group', 'mobile'],                    });                    renderData.accountList = accountList;                    renderData.accountGroup = accountGroup.map((item, idx) => {                        const groupList = accountList.filter(item => item.account_group === idx);                        return { groupName: item, groupList };                    });                }                await this.layout('change/apply_information.ejs', renderData, 'change/apply_information_modal.ejs');            } catch (err) {                this.log(err);                ctx.redirect('/tender/' + ctx.params.id + '/change');            }        }        async applyInformationNotice(ctx) {            try {                const tender = await ctx.service.tender.getDataById(ctx.tender.id);                // 获取附件列表                const fileList = await ctx.service.changeApplyAtt.getAllChangeApplyAtt(ctx.tender.id, ctx.change.id);                await this._getChangeApplyAuditViewData(ctx);                const renderData = {                    tender,                    change: ctx.change,                    changeConst,                    auditConst: audit.changeApply,                    fileList,                    returnUrl: this.app._.includes(ctx.request.headers.referer, '/tender/' + ctx.tender.id + '/change/apply') ? ctx.request.headers.referer : null,                    jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.change.apply_information_notice),                    preUrl: '/tender/' + ctx.tender.id + '/change/apply/' + ctx.change.id + '/information/notice',                };                await this.layout('change/apply_information_notice.ejs', renderData, 'change/apply_information_notice_modal.ejs');            } catch (err) {                this.log(err);                ctx.redirect('/tender/' + ctx.params.id + '/change');            }        }        // 审批相关        /**         * 添加审批人         * @param ctx         * @return {Promise<void>}         */        async addApplyAudit(ctx) {            try {                const auditConst = audit.changeApply;                const data = JSON.parse(ctx.request.body.data);                const id = this.app._.toInteger(data.auditorId);                if (isNaN(id) || id <= 0) {                    throw '参数错误';                }                // 检查权限等                if (ctx.change.uid !== ctx.session.sessionUser.accountId) {                    throw '您无权添加审核人';                }                if (ctx.change.status === auditConst.status.checking || ctx.change.status === auditConst.status.checked) {                    throw '当前不允许添加审核人';                }                ctx.change.auditorList = await ctx.service.changeApplyAudit.getAuditors(ctx.change.id, ctx.change.times);                // 检查审核人是否已存在                const exist = this.app._.find(ctx.change.auditorList, { aid: id });                if (exist) {                    throw '该审核人已存在,请勿重复添加';                }                // const shenpiInfo = await ctx.service.shenpiAudit.getDataByCondition({ tid: ctx.tender.id, sp_type: shenpiConst.sp_type.material, sp_status: shenpiConst.sp_status.gdzs });                // const is_gdzs = shenpiInfo && ctx.tender.info.shenpi.material === shenpiConst.sp_status.gdzs ? 1 : 0;                const result = await ctx.service.changeApplyAudit.addAuditor(ctx.change.id, id, ctx.change.times);                if (!result) {                    throw '添加审核人失败';                }                const auditors = await ctx.service.changeApplyAudit.getAuditorsWithOwner(ctx.change.id, ctx.change.times);                ctx.body = { err: 0, msg: '', data: auditors };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 移除审批人         * @param ctx         * @return {Promise<void>}         */        async deleteApplyAudit(ctx) {            try {                const data = JSON.parse(ctx.request.body.data);                const id = data.auditorId instanceof Number ? data.auditorId : this.app._.toNumber(data.auditorId);                if (isNaN(id) || id <= 0) {                    throw '参数错误';                }                const result = await ctx.service.changeApplyAudit.deleteAuditor(ctx.change.id, id, ctx.change.times);                if (!result) {                    throw '移除审核人失败';                }                const auditors = await ctx.service.changeApplyAudit.getAuditors(ctx.change.id, ctx.change.times);                ctx.body = { err: 0, msg: '', data: auditors };            } catch (err) {                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 上传附件         * @param {*} ctx 上下文         */        async uploadApplyFile(ctx) {            let stream;            try {                const auditConst = audit.changeApply;                // this._checkAdvanceFileCanModify(ctx);                const parts = this.ctx.multipart({                    autoFields: true,                });                const files = [];                const create_time = Date.parse(new Date()) / 1000;                let idx = 0;                const extra_upload = ctx.change.status === auditConst.status.checked;                while ((stream = await parts()) !== undefined) {                    if (!stream.filename) {                        // 如果没有传入直接返回                        return;                    }                    const fileInfo = path.parse(stream.filename);                    const filepath = `app/public/upload/${this.ctx.tender.id.toString()}/change_apply/fujian_${create_time + idx.toString() + fileInfo.ext}`;                    // await ctx.helper.saveStreamFile(stream, path.resolve(this.app.baseDir, 'app', filepath));                    await ctx.app.fujianOss.put(ctx.app.config.fujianOssFolder + filepath, stream);                    files.push({ filepath, name: stream.filename, ext: fileInfo.ext });                    ++idx;                    stream && (await sendToWormhole(stream));                }                const in_time = new Date();                const payload = files.map(file => {                    let idx;                    if (Array.isArray(parts.field.name)) {                        idx = parts.field.name.findIndex(name => name === file.name);                    } else {                        idx = 'isString';                    }                    const newFile = {                        tid: ctx.tender.id,                        caid: ctx.change.id,                        uid: ctx.session.sessionUser.accountId,                        type: parts.field.type,                        filename: file.name,                        fileext: file.ext,                        filesize: ctx.helper.bytesToSize(idx === 'isString' ? parts.field.size : parts.field.size[idx]),                        filepath: file.filepath,                        upload_time: in_time,                        extra_upload,                    };                    return newFile;                });                // 执行文件信息写入数据库                await ctx.service.changeApplyAtt.saveFileMsgToDb(payload);                // 将最新的当前标段的所有文件信息返回                const data = await ctx.service.changeApplyAtt.getAllChangeApplyAtt(ctx.tender.id, ctx.change.id);                ctx.body = { err: 0, msg: '', data };            } catch (err) {                stream && (await sendToWormhole(stream));                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 删除附件         * @param {Ojbect} ctx 上下文         */        async deleteApplyFile(ctx) {            try {                const { id } = JSON.parse(ctx.request.body.data);                const fileInfo = await ctx.service.changeApplyAtt.getDataById(id);                if (fileInfo || Object.keys(fileInfo).length) {                    // 先删除文件                    // await fs.unlinkSync(path.resolve(this.app.baseDir, './app', fileInfo.filepath));                    await ctx.app.fujianOss.delete(ctx.app.config.fujianOssFolder + fileInfo.filepath);                    // 再删除数据库                    await ctx.service.changeApplyAtt.delete(id);                } else {                    throw '不存在该文件';                }                const data = await ctx.service.changeApplyAtt.getAllChangeApplyAtt(ctx.tender.id, ctx.change.id);                ctx.body = { err: 0, msg: '请求成功', data };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 下载附件         * @param {Object} ctx - egg全局变量         * @return {void}         */        async downloadApplyFile(ctx) {            const id = ctx.params.fid;            if (id) {                try {                    const fileInfo = await ctx.service.changeApplyAtt.getDataById(id);                    if (fileInfo !== undefined && fileInfo !== '') {                        // const fileName = path.join(__dirname, '../', fileInfo.filepath);                        // 解决中文无法下载问题                        const userAgent = (ctx.request.header['user-agent'] || '').toLowerCase();                        let disposition = '';                        if (userAgent.indexOf('msie') >= 0 || userAgent.indexOf('chrome') >= 0) {                            disposition = 'attachment; filename=' + encodeURIComponent(fileInfo.filename);                        } else if (userAgent.indexOf('firefox') >= 0) {                            disposition = 'attachment; filename*="utf8\'\'' + encodeURIComponent(fileInfo.filename) + '"';                        } else {                            /* safari等其他非主流浏览器只能自求多福了 */                            disposition = 'attachment; filename=' + new Buffer(fileInfo.filename).toString('binary');                        }                        ctx.response.set({                            'Content-Type': 'application/octet-stream',                            'Content-Disposition': disposition,                            'Content-Length': fileInfo.filesize,                        });                        // ctx.body = await fs.createReadStream(fileName);                        ctx.body = await ctx.helper.ossFileGet(fileInfo.filepath);                    } else {                        throw '不存在该文件';                    }                } catch (err) {                    this.log(err);                    this.setMessage(err.toString(), this.messageType.ERROR);                }            }        }        async applyInformationSave(ctx) {            try {                const data = JSON.parse(ctx.request.body.data);                if (data.name === 'code') {                    const info = await ctx.service.changeApply.isRepeat(ctx.change.id, data.val, ctx.tender.id);                    if (info) {                        throw '该编号已存在';                    }                }                const result = await ctx.service.changeApply.saveInfo(ctx.change.id, data);                if (!result) {                    throw '修改失败';                }                ctx.body = { err: 0, msg: '请求成功', data: null };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 上报和重新上报         * @param ctx         * @return {Promise<void>}         */        async startApplyAudit(ctx) {            try {                const auditConst = audit.changeApply;                // 检查权限等                if (!ctx.change) {                    throw '数据错误';                }                if (ctx.change.uid !== ctx.session.sessionUser.accountId) {                    throw '您无权上报该期数据';                }                if (ctx.change.status === auditConst.status.checking || ctx.change.status === auditConst.status.checked) {                    throw '该材料调差期数据当前无法上报';                }                await ctx.service.changeApplyAudit.start(ctx.change.id, ctx.change.times);                ctx.redirect(ctx.request.header.referer);            } catch (err) {                this.log(err);                ctx.session.postError = err.toString();                ctx.redirect(ctx.request.header.referer);            }        }        /**         * 审批         * @param ctx         * @return {Promise<void>}         */        async checkApplyAudit(ctx) {            try {                const auditConst = audit.changeApply;                if (!ctx.change || ctx.change.status !== auditConst.status.checking) {                    throw '当前材料调差期数据有误';                }                if (!ctx.change.curAuditor || ctx.change.curAuditor.aid !== ctx.session.sessionUser.accountId) {                    throw '您无权进行该操作';                }                const data = {                    checkType: parseInt(ctx.request.body.checkType),                    opinion: ctx.request.body.opinion,                };                if (!data.checkType || isNaN(data.checkType)) {                    throw '提交数据错误';                }                if (ctx.request.body.notice_uid) {                    data.notice_code = ctx.request.body.notice_code;                    data.notice_uid = ctx.request.body.notice_uid;                }                // if (data.checkType === auditConst.status.checkNo) {                //     if (!data.checkType || isNaN(data.checkType)) {                //         throw '提交数据错误';                //     }                // }                await ctx.service.changeApplyAudit.check(ctx.change.id, data, ctx.change.times);                ctx.redirect(ctx.request.header.referer);            } catch (err) {                this.log(err);                ctx.session.postError = err.toString();                ctx.redirect(ctx.request.header.referer);            }        }        /**         * 变更清单 - 操作 (Ajax)         * @param ctx         * @return {Promise<void>}         */        async saveApplyListsData(ctx) {            try {                const data = JSON.parse(ctx.request.body.data);                const responseData = {                    err: 0,                    msg: '',                    data: {},                };                switch (data.type) {                    case 'add':                        responseData.data = await ctx.service.changeApplyList.add(data.updateData);                        break;                    case 'batchadd':                        responseData.data = await ctx.service.changeApplyList.batchAdd(data);                        break;                    case 'del':                        await ctx.service.changeApplyList.del(data.ids);                        // 取所有工料表                        responseData.data = await ctx.service.changeApplyList.getList(ctx.change.id);                        break;                    case 'update':                        // if (data.updateData.code === '' || data.updateData.code === null) {                        //     throw '请先输入编号';                        // }                        await ctx.service.changeApplyList.save(data.updateData);                        break;                    case 'paste':                        await ctx.service.changeApplyList.saveDatas(data.updateData);                        // 取所有工料表                        responseData.data = await ctx.service.changeApplyList.getList(ctx.change.id);                        break;                    case 'list_rule':                        const result = await ctx.service.tender.saveTenderData(ctx.tender.id, { c_apply_list_rule: data.postData });                        if (!result) {                            throw '修改失败';                        }                        break;                    // case 'update_tp':                    //     await ctx.service.changeApply.saveInfo(ctx.change.id, { name: 'total_price', val: data.updateData });                    //     break;                    default: throw '参数有误';                }                ctx.body = responseData;            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        // 变更方案        async _filterChangesPlan(ctx, status = 0) {            const tenderId = ctx.params.id;            ctx.session.sessionUser.tenderId = tenderId;            const tender = await this.service.tender.getDataById(ctx.tender.id);            // const tender = ctx.tender;            // const tenderList = await this.service.tender.getList();            const page = ctx.page;            const sorts = ctx.query.sort ? ctx.query.sort : 0;            const orders = ctx.query.order ? ctx.query.order : 0;            const changes = await ctx.service.changePlan.getListByStatus(tender.id, status, 1, sorts, orders);            const total = await ctx.service.changePlan.getCountByStatus(tender.id, status);            let page_total = 0;            const tp = await ctx.service.changePlan.getTp(tender.id, status);            for (const c of changes) {                c.curAuditor = await ctx.service.changePlanAudit.getAuditorByStatus(c.id, c.status, c.times);                page_total = ctx.helper.add(page_total, c.total_price);            }            const tender_userInfo = await ctx.service.projectAccount.getDataById(ctx.tender.data.user_id);            // 分页相关            const pageInfo = {                page,                total: Math.ceil(total / app.config.pageSize),                queryData: JSON.stringify(ctx.urlInfo.query),            };            const filter = JSON.parse(JSON.stringify(audit.changePlan.filter));            filter.count = [];            filter.count[filter.status.pending] = await ctx.service.changePlan.getCountByStatus(tender.id, filter.status.pending);// await ctx.service.change.pendingDatas(tender.id, ctx.session.sessionUser.accountId);            filter.count[filter.status.uncheck] = await ctx.service.changePlan.getCountByStatus(tender.id, filter.status.uncheck);// await ctx.service.change.checkingDatas(tender.id, ctx.session.sessionUser.accountId);            filter.count[filter.status.checking] = await ctx.service.changePlan.getCountByStatus(tender.id, filter.status.checking);// await ctx.service.change.checkedDatas(tender.id, ctx.session.sessionUser.accountId);            filter.count[filter.status.checked] = await ctx.service.changePlan.getCountByStatus(tender.id, filter.status.checked);// await ctx.service.change.pendingDatas(tender.id, ctx.session.sessionUser.accountId);            // filter.count[filter.status.checkNo] = await ctx.service.changeApply.getCountByStatus(tender.id, filter.status.checkNo);// await ctx.service.change.pendingDatas(tender.id, ctx.session.sessionUser.accountId);            let codeRule = [];            let c_connector = '1';            let c_rule_first = 1;            const rule_type = 'plan';            if (tender.c_code_rules) {                const c_code_rules = JSON.parse(tender.c_code_rules);                codeRule = c_code_rules[rule_type + '_rule'] !== undefined ? c_code_rules[rule_type + '_rule'] : [];                c_connector = c_code_rules[rule_type + '_connector'] !== undefined ? c_code_rules[rule_type + '_connector'] : '1';                c_rule_first = c_code_rules[rule_type + '_rule_first'] !== undefined ? c_code_rules[rule_type + '_rule_first'] : 1;            }            for (const rule of codeRule) {                switch (rule.rule_type) {                    case codeRuleConst.measure.ruleType.dealCode:                        rule.preview = ctx.tender.info.deal_info.dealCode;                        break;                    case codeRuleConst.measure.ruleType.tenderName:                        rule.preview = tender.name;                        break;                    case codeRuleConst.measure.ruleType.inDate:                        rule.preview = moment().format('YYYY');                        break;                    case codeRuleConst.measure.ruleType.text:                        rule.preview = rule.text;                        break;                    case codeRuleConst.measure.ruleType.addNo:                        const s = '0000000000';                        rule.preview = s.substr(s.length - rule.format);                        break;                    default: break;                }            }            const changeApplyList = await ctx.service.changeApply.getAllDataByCondition({ where: { tid: tender.id, status: audit.changeApply.status.checked } });            const allApplyCodes = await ctx.service.changePlan.getAllDataByCondition({                columns: ['apply_code'],                where: {                    tid: tender.id,                },            });            const acLists = allApplyCodes.length > 0 ? ctx.app._.uniq(ctx.app._.map(allApplyCodes, 'apply_code')) : [];            const renderData = {                uid: ctx.session.sessionUser.accountId,                tender,                pageInfo,                changes,                status,                rule_type,                codeRule,                c_connector,                c_rule_first,                filter,                plan_username: tender_userInfo ? tender_userInfo.name : null,                ruleType: codeRuleConst.ruleType[rule_type],                dealCode: ctx.tender.info.deal_info.dealCode,                auditConst: audit.changePlan,                ruleConst: codeRuleConst.measure,                changeConst,                changeApplyList,                acLists,                page_total,                tp,                tenderMenu: this.menu.tenderMenu,                preUrl: '/tender/' + tender.id,                jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.change.plan),            };            await this.layout('change/plan.ejs', renderData, 'change/plan_modal.ejs');        }        /**         * 变更申请列表 页面 (Get)         *         * @param {Object} ctx - egg全局变量         * @return {void}         */        async plan(ctx) {            try {                await this._filterChangesPlan(ctx);            } catch (err) {                this.log(err);                ctx.redirect('/dashboard');            }        }        /**         * 变更管理 状态筛选 页面 (Get)         * @param {Object} ctx - egg全局变量         * @return {void}         */        async planStatus(ctx) {            try {                const status = parseInt(ctx.params.status);                await this._filterChangesPlan(ctx, status);            } catch (err) {                this.logger.error(err);                ctx.redirect('/tender/' + ctx.params.id + '/change/plan');            }        }        /**         * 新增变更立项 (Post)         *         * @param {Object} ctx - egg全局变量         * @return {void}         */        async planAdd(ctx) {            try {                const tenderId = ctx.params.id;                if (!tenderId) {                    throw '当前未打开标段';                }                const data = JSON.parse(ctx.request.body.data);                if (!data.code || data.code === '') {                    throw '变更方案编号不能为空';                }                const change = await ctx.service.changePlan.add(tenderId, ctx.session.sessionUser.accountId, data.code, data.apply_code, data.name);                ctx.body = { err: 0, msg: '', data: change };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString() };            }        }        /**         * 删除变更立项         * @param {Object} ctx - egg全局变量         * @return {void}         */        async planDelete(ctx) {            try {                const result = await ctx.service.changePlan.delete(ctx.request.body.cpid);                if (!result) {                    throw '删除变更方案失败';                }                ctx.redirect(ctx.request.header.referer);            } catch (err) {                console.log(err);                ctx.redirect(ctx.request.header.referer);            }        }        /**         * 获取审批界面所需的 原报、审批人数据等         * @param ctx         * @return {Promise<void>}         * @private         */        async _getChangePlanAuditViewData(ctx) {            const auditConst = audit.changePlan;            const times = ctx.change.status === auditConst.status.checkNo ? ctx.change.times - 1 : ctx.change.times;            ctx.change.user = await ctx.service.projectAccount.getAccountInfoById(ctx.change.uid);            ctx.change.auditHistory = [];            if (times >= 1) {                for (let i = 1; i <= times; i++) {                    ctx.change.auditHistory.push(await ctx.service.changePlanAudit.getAuditors(ctx.change.id, i));                }            }            // 获取审批流程中左边列表            ctx.change.auditors2 = ctx.change.status === auditConst.status.checkNo && ctx.change.uid !== ctx.session.sessionUser.accountId ?                await ctx.service.changePlanAudit.getAuditorsWithOwner(ctx.change.id, times) :                await ctx.service.changePlanAudit.getAuditorsWithOwner(ctx.change.id, ctx.change.times);            if (ctx.change.status === auditConst.status.uncheck || ctx.change.status === auditConst.status.checkNo) {                ctx.change.auditorList = await ctx.service.changePlanAudit.getAuditors(ctx.change.id, ctx.change.times);            }        }        async planInformation(ctx) {            try {                const whiteList = this.ctx.app.config.multipart.whitelist;                const tender = await ctx.service.tender.getDataById(ctx.tender.id);                await this._getChangePlanAuditViewData(ctx);                // 获取附件列表                const fileList = await ctx.service.changePlanAtt.getAllChangePlanAtt(ctx.tender.id, ctx.change.id);                // 获取清单列表                const changeList = await ctx.service.changePlanList.getList(ctx.change.id);                if (ctx.change.status === audit.changePlan.status.checking || ctx.change.status === audit.changePlan.status.checked) {                    const listAudits = await ctx.service.changePlanAudit.getAuditGroupByList(ctx.change.id, ctx.change.times);                    ctx.change.listAudits = listAudits;                    for (const cl of changeList) {                        const audit_amount = cl.audit_amount !== null && cl.audit_amount !== '' ? cl.audit_amount.split(',') : '';                        // 清单表页赋值                        for (const [index, au] of listAudits.entries()) {                            cl['audit_amount_' + au.aid] = ctx.change.shenpiPower && au.aid === ctx.session.sessionUser.accountId ? cl.spamount : (audit_amount[index] ? parseFloat(audit_amount[index]) : null);                        }                    }                }                const renderData = {                    tender,                    change: ctx.change,                    listRule: tender.c_plan_list_rule ? JSON.parse(tender.c_plan_list_rule) : { source: 1, rule: ['unit', 'unit_price'] },                    changeList,                    changeConst,                    auditConst: audit.changePlan,                    fileList,                    whiteList,                    tpUnit: ctx.change.decimal ? ctx.change.decimal.tp : ctx.tender.info.decimal.tp,                    upUnit: ctx.change.decimal ? ctx.change.decimal.up : ctx.tender.info.decimal.up,                    changeUnits: changeConst.units,                    precision: ctx.change.decimal && ctx.change.decimal.precision ? ctx.change.decimal.precision : ctx.tender.info.precision,                    returnUrl: this.app._.includes(ctx.request.headers.referer, '/tender/' + ctx.tender.id + '/change/plan') && !this.app._.includes(ctx.request.headers.referer, '/tender/' + ctx.tender.id + '/change/plan/' + ctx.change.id + '/information') ? ctx.request.headers.referer : null,                    jsFiles: this.app.jsFiles.common.concat(this.app.jsFiles.change.plan_information),                    preUrl: '/tender/' + ctx.tender.id + '/change/plan/' + ctx.change.id + '/information',                };                if ((ctx.change.status === audit.changePlan.status.uncheck || ctx.change.status === audit.changePlan.status.checkNo) && (ctx.session.sessionUser.accountId === ctx.change.uid || ctx.tender.isTourist)) {                    // data.accountGroup = accountGroup;                    // 获取所有项目参与者                    const accountList = await ctx.service.projectAccount.getAllDataByCondition({                        where: { project_id: ctx.session.sessionProject.id, enable: 1 },                        columns: ['id', 'name', 'company', 'role', 'enable', 'is_admin', 'account_group', 'mobile'],                    });                    renderData.accountList = accountList;                    renderData.accountGroup = accountGroup.map((item, idx) => {                        const groupList = accountList.filter(item => item.account_group === idx);                        return { groupName: item, groupList };                    });                }                await this.layout('change/plan_information.ejs', renderData, 'change/plan_information_modal.ejs');            } catch (err) {                this.log(err);                ctx.redirect('/tender/' + ctx.params.id + '/change');            }        }        // 审批相关        /**         * 添加审批人         * @param ctx         * @return {Promise<void>}         */        async addPlanAudit(ctx) {            try {                const auditConst = audit.changePlan;                const data = JSON.parse(ctx.request.body.data);                const id = this.app._.toInteger(data.auditorId);                if (isNaN(id) || id <= 0) {                    throw '参数错误';                }                // 检查权限等                if (ctx.change.uid !== ctx.session.sessionUser.accountId) {                    throw '您无权添加审核人';                }                if (ctx.change.status === auditConst.status.checking || ctx.change.status === auditConst.status.checked) {                    throw '当前不允许添加审核人';                }                ctx.change.auditorList = await ctx.service.changePlanAudit.getAuditors(ctx.change.id, ctx.change.times);                // 检查审核人是否已存在                const exist = this.app._.find(ctx.change.auditorList, { aid: id });                if (exist) {                    throw '该审核人已存在,请勿重复添加';                }                // const shenpiInfo = await ctx.service.shenpiAudit.getDataByCondition({ tid: ctx.tender.id, sp_type: shenpiConst.sp_type.material, sp_status: shenpiConst.sp_status.gdzs });                // const is_gdzs = shenpiInfo && ctx.tender.info.shenpi.material === shenpiConst.sp_status.gdzs ? 1 : 0;                const result = await ctx.service.changePlanAudit.addAuditor(ctx.change.id, id, ctx.change.times);                if (!result) {                    throw '添加审核人失败';                }                const auditors = await ctx.service.changePlanAudit.getAuditorsWithOwner(ctx.change.id, ctx.change.times);                ctx.body = { err: 0, msg: '', data: auditors };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 移除审批人         * @param ctx         * @return {Promise<void>}         */        async deletePlanAudit(ctx) {            try {                const data = JSON.parse(ctx.request.body.data);                const id = data.auditorId instanceof Number ? data.auditorId : this.app._.toNumber(data.auditorId);                if (isNaN(id) || id <= 0) {                    throw '参数错误';                }                const result = await ctx.service.changePlanAudit.deleteAuditor(ctx.change.id, id, ctx.change.times);                if (!result) {                    throw '移除审核人失败';                }                const auditors = await ctx.service.changePlanAudit.getAuditors(ctx.change.id, ctx.change.times);                ctx.body = { err: 0, msg: '', data: auditors };            } catch (err) {                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 上传附件         * @param {*} ctx 上下文         */        async uploadPlanFile(ctx) {            let stream;            try {                const auditConst = audit.changePlan;                // this._checkAdvanceFileCanModify(ctx);                const parts = this.ctx.multipart({                    autoFields: true,                });                const files = [];                const create_time = Date.parse(new Date()) / 1000;                let idx = 0;                const extra_upload = ctx.change.status === auditConst.status.checked;                while ((stream = await parts()) !== undefined) {                    if (!stream.filename) {                        // 如果没有传入直接返回                        return;                    }                    const fileInfo = path.parse(stream.filename);                    const filepath = `app/public/upload/${this.ctx.tender.id.toString()}/change_plan/fujian_${create_time + idx.toString() + fileInfo.ext}`;                    // await ctx.helper.saveStreamFile(stream, path.resolve(this.app.baseDir, 'app', filepath));                    await ctx.app.fujianOss.put(ctx.app.config.fujianOssFolder + filepath, stream);                    files.push({ filepath, name: stream.filename, ext: fileInfo.ext });                    ++idx;                    stream && (await sendToWormhole(stream));                }                const in_time = new Date();                const payload = files.map(file => {                    let idx;                    if (Array.isArray(parts.field.name)) {                        idx = parts.field.name.findIndex(name => name === file.name);                    } else {                        idx = 'isString';                    }                    const newFile = {                        tid: ctx.tender.id,                        cpid: ctx.change.id,                        uid: ctx.session.sessionUser.accountId,                        filename: file.name,                        fileext: file.ext,                        filesize: ctx.helper.bytesToSize(idx === 'isString' ? parts.field.size : parts.field.size[idx]),                        filepath: file.filepath,                        upload_time: in_time,                        extra_upload,                    };                    return newFile;                });                // 执行文件信息写入数据库                await ctx.service.changePlanAtt.saveFileMsgToDb(payload);                // 将最新的当前标段的所有文件信息返回                const data = await ctx.service.changePlanAtt.getAllChangePlanAtt(ctx.tender.id, ctx.change.id);                ctx.body = { err: 0, msg: '', data };            } catch (err) {                stream && (await sendToWormhole(stream));                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 删除附件         * @param {Ojbect} ctx 上下文         */        async deletePlanFile(ctx) {            try {                const { id } = JSON.parse(ctx.request.body.data);                const fileInfo = await ctx.service.changePlanAtt.getDataById(id);                if (fileInfo || Object.keys(fileInfo).length) {                    // 先删除文件                    // await fs.unlinkSync(path.resolve(this.app.baseDir, './app', fileInfo.filepath));                    await ctx.app.fujianOss.delete(ctx.app.config.fujianOssFolder + fileInfo.filepath);                    // 再删除数据库                    await ctx.service.changePlanAtt.delete(id);                } else {                    throw '不存在该文件';                }                const data = await ctx.service.changePlanAtt.getAllChangePlanAtt(ctx.tender.id, ctx.change.id);                ctx.body = { err: 0, msg: '请求成功', data };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 下载附件         * @param {Object} ctx - egg全局变量         * @return {void}         */        async downloadPlanFile(ctx) {            const id = ctx.params.fid;            if (id) {                try {                    const fileInfo = await ctx.service.changePlanAtt.getDataById(id);                    if (fileInfo !== undefined && fileInfo !== '') {                        // const fileName = path.join(__dirname, '../', fileInfo.filepath);                        // 解决中文无法下载问题                        const userAgent = (ctx.request.header['user-agent'] || '').toLowerCase();                        let disposition = '';                        if (userAgent.indexOf('msie') >= 0 || userAgent.indexOf('chrome') >= 0) {                            disposition = 'attachment; filename=' + encodeURIComponent(fileInfo.filename);                        } else if (userAgent.indexOf('firefox') >= 0) {                            disposition = 'attachment; filename*="utf8\'\'' + encodeURIComponent(fileInfo.filename) + '"';                        } else {                            /* safari等其他非主流浏览器只能自求多福了 */                            disposition = 'attachment; filename=' + new Buffer(fileInfo.filename).toString('binary');                        }                        ctx.response.set({                            'Content-Type': 'application/octet-stream',                            'Content-Disposition': disposition,                            'Content-Length': fileInfo.filesize,                        });                        // ctx.body = await fs.createReadStream(fileName);                        ctx.body = await ctx.helper.ossFileGet(fileInfo.filepath);                    } else {                        throw '不存在该文件';                    }                } catch (err) {                    this.log(err);                    this.setMessage(err.toString(), this.messageType.ERROR);                }            }        }        async planInformationSave(ctx) {            try {                const data = JSON.parse(ctx.request.body.data);                if (data.name === 'code') {                    const info = await ctx.service.changePlan.isRepeat(ctx.change.id, data.val, ctx.tender.id);                    if (info) {                        throw '该编号已存在';                    }                }                const result = await ctx.service.changePlan.saveInfo(ctx.change.id, data);                if (!result) {                    throw '修改失败';                }                ctx.body = { err: 0, msg: '请求成功', data: null };            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }        /**         * 上报和重新上报         * @param ctx         * @return {Promise<void>}         */        async startPlanAudit(ctx) {            try {                const auditConst = audit.changePlan;                // 检查权限等                if (!ctx.change) {                    throw '数据错误';                }                if (ctx.change.uid !== ctx.session.sessionUser.accountId) {                    throw '您无权上报该期数据';                }                if (ctx.change.status === auditConst.status.checking || ctx.change.status === auditConst.status.checked) {                    throw '该材料调差期数据当前无法上报';                }                await ctx.service.changePlanAudit.start(ctx.change.id, ctx.change.times);                ctx.redirect(ctx.request.header.referer);            } catch (err) {                this.log(err);                ctx.session.postError = err.toString();                ctx.redirect(ctx.request.header.referer);            }        }        /**         * 审批         * @param ctx         * @return {Promise<void>}         */        async checkPlanAudit(ctx) {            try {                const auditConst = audit.changePlan;                if (!ctx.change || ctx.change.status !== auditConst.status.checking) {                    throw '当前材料调差期数据有误';                }                if (!ctx.change.curAuditor || ctx.change.curAuditor.aid !== ctx.session.sessionUser.accountId) {                    throw '您无权进行该操作';                }                const data = {                    checkType: parseInt(ctx.request.body.checkType),                    opinion: ctx.request.body.opinion,                };                if (!data.checkType || isNaN(data.checkType)) {                    throw '提交数据错误';                }                // if (data.checkType === auditConst.status.checkNo) {                //     if (!data.checkType || isNaN(data.checkType)) {                //         throw '提交数据错误';                //     }                // }                await ctx.service.changePlanAudit.check(ctx.change.id, data, ctx.change.times);                ctx.redirect(ctx.request.header.referer);            } catch (err) {                this.log(err);                console.log(err);                ctx.session.postError = err.toString();                ctx.redirect(ctx.request.header.referer);            }        }        /**         * 变更清单 - 操作 (Ajax)         * @param ctx         * @return {Promise<void>}         */        async savePlanListsData(ctx) {            try {                const data = JSON.parse(ctx.request.body.data);                const responseData = {                    err: 0,                    msg: '',                    data: {},                };                switch (data.type) {                    case 'add':                        responseData.data = await ctx.service.changePlanList.add(data.updateData);                        break;                    case 'batchadd':                        responseData.data = await ctx.service.changePlanList.batchAdd(data);                        break;                    case 'del':                        await ctx.service.changePlanList.del(data.ids);                        // 取所有工料表                        responseData.data = await ctx.service.changePlanList.getList(ctx.change.id);                        break;                    case 'update':                        // if (data.updateData.code === '' || data.updateData.code === null) {                        //     throw '请先输入编号';                        // }                        await ctx.service.changePlanList.save(data.updateData);                        break;                    case 'paste':                        await ctx.service.changePlanList.saveDatas(data.updateData);                        // 取所有工料表                        responseData.data = await ctx.service.changePlanList.getList(ctx.change.id);                        if (ctx.change.status === audit.changePlan.status.checking || ctx.change.status === audit.changePlan.status.checked) {                            const listAudits = await ctx.service.changePlanAudit.getAuditGroupByList(ctx.change.id, ctx.change.times);                            for (const cl of responseData.data) {                                const audit_amount = cl.audit_amount !== null && cl.audit_amount !== '' ? cl.audit_amount.split(',') : '';                                // 清单表页赋值                                for (const [index, au] of listAudits.entries()) {                                    cl['audit_amount_' + au.aid] = ctx.change.shenpiPower && au.aid === ctx.session.sessionUser.accountId ? cl.spamount : (audit_amount[index] ? parseFloat(audit_amount[index]) : null);                                }                            }                        }                        break;                    case 'list_rule':                        const result = await ctx.service.tender.saveTenderData(ctx.tender.id, { c_plan_list_rule: data.postData });                        if (!result) {                            throw '修改失败';                        }                        break;                    case 'paste_amount_rows':                        await ctx.service.changePlanList.saveDatas(data.updateData);                        const changeList = await ctx.service.changePlanList.getList(ctx.change.id);                        const listAudits = await ctx.service.changePlanAudit.getAuditGroupByList(ctx.change.id, ctx.change.times);                        for (const cl of changeList) {                            // 清单表页赋值                            const audit_amount = cl.audit_amount !== null && cl.audit_amount !== '' ? cl.audit_amount.split(',') : '';                            // 清单表页赋值                            for (const [index, au] of listAudits.entries()) {                                cl['audit_amount_' + au.aid] = ctx.change.shenpiPower && au.aid === ctx.session.sessionUser.accountId ? cl.spamount : (audit_amount[index] ? parseFloat(audit_amount[index]) : null);                            }                        }                        // 取所有工料表                        responseData.data = changeList;                        break;                    // case 'update_tp':                    //     await ctx.service.changePlan.saveInfo(ctx.change.id, { name: 'total_price', val: data.updateData });                    //     break;                    default: throw '参数有误';                }                ctx.body = responseData;            } catch (err) {                this.log(err);                ctx.body = { err: 1, msg: err.toString(), data: null };            }        }    }    return ChangeController;};
 |