cos-js-sdk-v5.js 345 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639664066416642664366446645664666476648664966506651665266536654665566566657665866596660666166626663666466656666666766686669667066716672667366746675667666776678667966806681668266836684668566866687668866896690669166926693669466956696669766986699670067016702670367046705670667076708670967106711671267136714671567166717671867196720672167226723672467256726672767286729673067316732673367346735673667376738673967406741674267436744674567466747674867496750675167526753675467556756675767586759676067616762676367646765676667676768676967706771677267736774677567766777677867796780678167826783678467856786678767886789679067916792679367946795679667976798679968006801680268036804680568066807680868096810681168126813681468156816681768186819682068216822682368246825682668276828682968306831683268336834683568366837683868396840684168426843684468456846684768486849685068516852685368546855685668576858685968606861686268636864686568666867686868696870687168726873687468756876687768786879688068816882688368846885688668876888688968906891689268936894689568966897689868996900690169026903690469056906690769086909691069116912691369146915691669176918691969206921692269236924692569266927692869296930693169326933693469356936693769386939694069416942694369446945694669476948694969506951695269536954695569566957695869596960696169626963696469656966696769686969697069716972697369746975697669776978697969806981698269836984698569866987698869896990699169926993699469956996699769986999700070017002700370047005700670077008700970107011701270137014701570167017701870197020702170227023702470257026702770287029703070317032703370347035703670377038703970407041704270437044704570467047704870497050705170527053705470557056705770587059706070617062706370647065706670677068706970707071707270737074707570767077707870797080708170827083708470857086708770887089709070917092709370947095709670977098709971007101710271037104710571067107710871097110711171127113711471157116711771187119712071217122712371247125712671277128712971307131713271337134713571367137713871397140714171427143714471457146714771487149715071517152715371547155715671577158715971607161716271637164716571667167716871697170717171727173717471757176717771787179718071817182718371847185718671877188718971907191719271937194719571967197719871997200720172027203720472057206720772087209721072117212721372147215721672177218721972207221722272237224722572267227722872297230723172327233723472357236723772387239724072417242724372447245724672477248724972507251725272537254725572567257725872597260726172627263726472657266726772687269727072717272727372747275727672777278727972807281728272837284728572867287728872897290729172927293729472957296729772987299730073017302730373047305730673077308730973107311731273137314731573167317731873197320732173227323732473257326732773287329733073317332733373347335733673377338733973407341734273437344734573467347734873497350735173527353735473557356735773587359736073617362736373647365736673677368736973707371737273737374737573767377737873797380738173827383738473857386738773887389739073917392739373947395739673977398739974007401740274037404740574067407740874097410741174127413741474157416741774187419742074217422742374247425742674277428742974307431743274337434743574367437743874397440744174427443744474457446744774487449745074517452745374547455745674577458745974607461746274637464746574667467746874697470747174727473747474757476747774787479748074817482748374847485748674877488748974907491749274937494749574967497749874997500750175027503750475057506750775087509751075117512751375147515751675177518751975207521752275237524752575267527752875297530753175327533753475357536753775387539754075417542754375447545754675477548754975507551755275537554755575567557755875597560756175627563756475657566756775687569757075717572757375747575757675777578757975807581758275837584758575867587758875897590759175927593759475957596759775987599760076017602760376047605760676077608760976107611761276137614761576167617761876197620762176227623762476257626762776287629763076317632763376347635763676377638763976407641764276437644764576467647764876497650765176527653765476557656765776587659766076617662766376647665766676677668766976707671767276737674767576767677767876797680768176827683768476857686768776887689769076917692769376947695769676977698769977007701770277037704770577067707770877097710771177127713771477157716771777187719772077217722772377247725772677277728772977307731773277337734773577367737773877397740774177427743774477457746774777487749775077517752775377547755775677577758775977607761776277637764776577667767776877697770777177727773777477757776777777787779778077817782778377847785778677877788778977907791779277937794779577967797779877997800780178027803780478057806780778087809781078117812781378147815781678177818781978207821782278237824782578267827782878297830783178327833783478357836783778387839784078417842784378447845784678477848784978507851785278537854785578567857785878597860786178627863786478657866786778687869787078717872787378747875787678777878787978807881788278837884788578867887788878897890789178927893789478957896789778987899790079017902790379047905790679077908790979107911791279137914791579167917791879197920792179227923792479257926792779287929793079317932793379347935793679377938793979407941794279437944794579467947794879497950795179527953795479557956795779587959796079617962796379647965796679677968796979707971797279737974797579767977797879797980798179827983798479857986798779887989799079917992799379947995799679977998799980008001800280038004800580068007800880098010801180128013801480158016801780188019802080218022802380248025802680278028802980308031803280338034803580368037803880398040804180428043804480458046804780488049805080518052805380548055805680578058805980608061806280638064806580668067806880698070807180728073807480758076807780788079808080818082808380848085808680878088808980908091809280938094809580968097809880998100810181028103810481058106810781088109811081118112811381148115811681178118811981208121812281238124812581268127812881298130813181328133813481358136813781388139814081418142814381448145814681478148814981508151815281538154815581568157815881598160816181628163816481658166816781688169817081718172817381748175817681778178817981808181818281838184818581868187818881898190819181928193819481958196819781988199820082018202820382048205820682078208820982108211821282138214821582168217821882198220822182228223822482258226822782288229823082318232823382348235823682378238823982408241824282438244824582468247824882498250825182528253825482558256825782588259826082618262826382648265826682678268826982708271827282738274827582768277827882798280828182828283828482858286828782888289829082918292829382948295829682978298829983008301830283038304830583068307830883098310831183128313831483158316831783188319832083218322832383248325832683278328832983308331833283338334833583368337833883398340834183428343834483458346834783488349835083518352835383548355835683578358835983608361836283638364836583668367836883698370837183728373837483758376837783788379838083818382838383848385838683878388838983908391839283938394839583968397839883998400840184028403840484058406840784088409841084118412841384148415841684178418841984208421842284238424842584268427842884298430843184328433843484358436843784388439844084418442844384448445844684478448844984508451845284538454845584568457845884598460846184628463846484658466846784688469847084718472847384748475847684778478847984808481848284838484848584868487848884898490849184928493849484958496849784988499850085018502850385048505850685078508850985108511851285138514851585168517851885198520852185228523852485258526852785288529853085318532853385348535853685378538853985408541854285438544854585468547854885498550855185528553855485558556855785588559856085618562856385648565856685678568856985708571857285738574857585768577857885798580858185828583858485858586858785888589859085918592859385948595859685978598859986008601860286038604860586068607860886098610861186128613861486158616861786188619862086218622862386248625862686278628862986308631863286338634863586368637863886398640864186428643864486458646864786488649865086518652865386548655865686578658865986608661866286638664866586668667866886698670867186728673867486758676867786788679868086818682868386848685868686878688868986908691869286938694869586968697869886998700870187028703870487058706870787088709871087118712871387148715871687178718871987208721872287238724872587268727872887298730873187328733873487358736873787388739874087418742874387448745874687478748874987508751875287538754875587568757875887598760876187628763876487658766876787688769877087718772877387748775877687778778877987808781878287838784878587868787878887898790879187928793879487958796879787988799880088018802880388048805880688078808880988108811881288138814881588168817881888198820882188228823882488258826882788288829883088318832883388348835883688378838883988408841884288438844884588468847884888498850885188528853885488558856885788588859886088618862886388648865886688678868886988708871887288738874887588768877887888798880888188828883888488858886888788888889889088918892889388948895889688978898889989008901890289038904890589068907890889098910891189128913891489158916891789188919892089218922892389248925892689278928892989308931893289338934893589368937893889398940894189428943894489458946894789488949895089518952895389548955895689578958895989608961896289638964896589668967896889698970897189728973897489758976897789788979898089818982898389848985898689878988898989908991899289938994899589968997899889999000900190029003900490059006900790089009901090119012901390149015901690179018901990209021902290239024902590269027902890299030903190329033903490359036903790389039904090419042904390449045904690479048904990509051905290539054905590569057905890599060906190629063906490659066906790689069907090719072907390749075907690779078907990809081908290839084908590869087908890899090909190929093909490959096909790989099910091019102910391049105910691079108910991109111911291139114911591169117911891199120912191229123912491259126912791289129913091319132913391349135913691379138913991409141914291439144914591469147914891499150915191529153915491559156915791589159916091619162916391649165916691679168916991709171917291739174917591769177917891799180918191829183918491859186918791889189919091919192919391949195919691979198919992009201920292039204920592069207920892099210921192129213921492159216921792189219922092219222922392249225922692279228922992309231923292339234923592369237923892399240924192429243924492459246924792489249925092519252925392549255925692579258925992609261926292639264926592669267926892699270927192729273927492759276927792789279928092819282928392849285928692879288928992909291929292939294929592969297929892999300930193029303930493059306930793089309931093119312931393149315931693179318931993209321932293239324932593269327932893299330933193329333933493359336933793389339934093419342934393449345934693479348934993509351935293539354935593569357935893599360936193629363936493659366936793689369937093719372937393749375937693779378937993809381938293839384938593869387938893899390939193929393939493959396939793989399940094019402940394049405940694079408940994109411941294139414941594169417941894199420942194229423942494259426942794289429943094319432943394349435943694379438943994409441944294439444944594469447944894499450945194529453945494559456945794589459946094619462946394649465946694679468946994709471947294739474947594769477947894799480948194829483948494859486948794889489949094919492949394949495949694979498949995009501950295039504950595069507950895099510951195129513951495159516951795189519952095219522952395249525952695279528952995309531953295339534953595369537953895399540954195429543954495459546954795489549955095519552955395549555955695579558955995609561956295639564956595669567956895699570957195729573957495759576957795789579958095819582958395849585958695879588958995909591959295939594959595969597959895999600960196029603960496059606960796089609961096119612961396149615961696179618961996209621962296239624962596269627962896299630963196329633963496359636963796389639964096419642964396449645964696479648964996509651965296539654965596569657965896599660966196629663966496659666966796689669967096719672967396749675967696779678967996809681968296839684968596869687968896899690969196929693969496959696969796989699970097019702970397049705970697079708970997109711971297139714971597169717971897199720972197229723972497259726972797289729973097319732973397349735973697379738973997409741974297439744974597469747974897499750975197529753975497559756975797589759976097619762976397649765976697679768976997709771977297739774977597769777977897799780978197829783978497859786978797889789979097919792979397949795979697979798979998009801980298039804
  1. (function webpackUniversalModuleDefinition(root, factory) {
  2. if(typeof exports === 'object' && typeof module === 'object')
  3. module.exports = factory();
  4. else if(typeof define === 'function' && define.amd)
  5. define([], factory);
  6. else if(typeof exports === 'object')
  7. exports["COS"] = factory();
  8. else
  9. root["COS"] = factory();
  10. })(typeof self !== 'undefined' ? self : this, function() {
  11. return /******/ (function(modules) { // webpackBootstrap
  12. /******/ // The module cache
  13. /******/ var installedModules = {};
  14. /******/
  15. /******/ // The require function
  16. /******/ function __webpack_require__(moduleId) {
  17. /******/
  18. /******/ // Check if module is in cache
  19. /******/ if(installedModules[moduleId]) {
  20. /******/ return installedModules[moduleId].exports;
  21. /******/ }
  22. /******/ // Create a new module (and put it into the cache)
  23. /******/ var module = installedModules[moduleId] = {
  24. /******/ i: moduleId,
  25. /******/ l: false,
  26. /******/ exports: {}
  27. /******/ };
  28. /******/
  29. /******/ // Execute the module function
  30. /******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
  31. /******/
  32. /******/ // Flag the module as loaded
  33. /******/ module.l = true;
  34. /******/
  35. /******/ // Return the exports of the module
  36. /******/ return module.exports;
  37. /******/ }
  38. /******/
  39. /******/
  40. /******/ // expose the modules object (__webpack_modules__)
  41. /******/ __webpack_require__.m = modules;
  42. /******/
  43. /******/ // expose the module cache
  44. /******/ __webpack_require__.c = installedModules;
  45. /******/
  46. /******/ // define getter function for harmony exports
  47. /******/ __webpack_require__.d = function(exports, name, getter) {
  48. /******/ if(!__webpack_require__.o(exports, name)) {
  49. /******/ Object.defineProperty(exports, name, {
  50. /******/ configurable: false,
  51. /******/ enumerable: true,
  52. /******/ get: getter
  53. /******/ });
  54. /******/ }
  55. /******/ };
  56. /******/
  57. /******/ // getDefaultExport function for compatibility with non-harmony modules
  58. /******/ __webpack_require__.n = function(module) {
  59. /******/ var getter = module && module.__esModule ?
  60. /******/ function getDefault() { return module['default']; } :
  61. /******/ function getModuleExports() { return module; };
  62. /******/ __webpack_require__.d(getter, 'a', getter);
  63. /******/ return getter;
  64. /******/ };
  65. /******/
  66. /******/ // Object.prototype.hasOwnProperty.call
  67. /******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
  68. /******/
  69. /******/ // __webpack_public_path__
  70. /******/ __webpack_require__.p = "/dist/";
  71. /******/
  72. /******/ // Load entry module and return exports
  73. /******/ return __webpack_require__(__webpack_require__.s = 5);
  74. /******/ })
  75. /************************************************************************/
  76. /******/ ([
  77. /* 0 */
  78. /***/ (function(module, exports, __webpack_require__) {
  79. "use strict";
  80. /* WEBPACK VAR INJECTION */(function(process) {
  81. var md5 = __webpack_require__(7);
  82. var CryptoJS = __webpack_require__(10);
  83. var xml2json = __webpack_require__(11);
  84. var json2xml = __webpack_require__(14);
  85. function camSafeUrlEncode(str) {
  86. return encodeURIComponent(str).replace(/!/g, '%21').replace(/'/g, '%27').replace(/\(/g, '%28').replace(/\)/g, '%29').replace(/\*/g, '%2A');
  87. }
  88. //测试用的key后面可以去掉
  89. var getAuth = function (opt) {
  90. opt = opt || {};
  91. var SecretId = opt.SecretId;
  92. var SecretKey = opt.SecretKey;
  93. var KeyTime = opt.KeyTime;
  94. var method = (opt.method || opt.Method || 'get').toLowerCase();
  95. var queryParams = clone(opt.Query || opt.params || {});
  96. var headers = clone(opt.Headers || opt.headers || {});
  97. var Key = opt.Key || '';
  98. var pathname;
  99. if (opt.UseRawKey) {
  100. pathname = opt.Pathname || opt.pathname || '/' + Key;
  101. } else {
  102. pathname = opt.Pathname || opt.pathname || Key;
  103. pathname.indexOf('/') !== 0 && (pathname = '/' + pathname);
  104. }
  105. if (!SecretId) throw new Error('missing param SecretId');
  106. if (!SecretKey) throw new Error('missing param SecretKey');
  107. var getObjectKeys = function (obj, forKey) {
  108. var list = [];
  109. for (var key in obj) {
  110. if (obj.hasOwnProperty(key)) {
  111. list.push(forKey ? camSafeUrlEncode(key).toLowerCase() : key);
  112. }
  113. }
  114. return list.sort(function (a, b) {
  115. a = a.toLowerCase();
  116. b = b.toLowerCase();
  117. return a === b ? 0 : a > b ? 1 : -1;
  118. });
  119. };
  120. var obj2str = function (obj) {
  121. var i, key, val;
  122. var list = [];
  123. var keyList = getObjectKeys(obj);
  124. for (i = 0; i < keyList.length; i++) {
  125. key = keyList[i];
  126. val = obj[key] === undefined || obj[key] === null ? '' : '' + obj[key];
  127. key = camSafeUrlEncode(key).toLowerCase();
  128. val = camSafeUrlEncode(val) || '';
  129. list.push(key + '=' + val);
  130. }
  131. return list.join('&');
  132. };
  133. // 签名有效起止时间
  134. var now = Math.round(getSkewTime(opt.SystemClockOffset) / 1000) - 1;
  135. var exp = now;
  136. var Expires = opt.Expires || opt.expires;
  137. if (Expires === undefined) {
  138. exp += 900; // 签名过期时间为当前 + 900s
  139. } else {
  140. exp += Expires * 1 || 0;
  141. }
  142. // 要用到的 Authorization 参数列表
  143. var qSignAlgorithm = 'sha1';
  144. var qAk = SecretId;
  145. var qSignTime = KeyTime || now + ';' + exp;
  146. var qKeyTime = KeyTime || now + ';' + exp;
  147. var qHeaderList = getObjectKeys(headers).join(';').toLowerCase();
  148. var qUrlParamList = getObjectKeys(queryParams).join(';').toLowerCase();
  149. // 签名算法说明文档:https://www.qcloud.com/document/product/436/7778
  150. // 步骤一:计算 SignKey
  151. var signKey = CryptoJS.HmacSHA1(qKeyTime, SecretKey).toString();
  152. // 步骤二:构成 FormatString
  153. var formatString = [method, pathname, obj2str(queryParams), obj2str(headers), ''].join('\n');
  154. // 步骤三:计算 StringToSign
  155. var stringToSign = ['sha1', qSignTime, CryptoJS.SHA1(formatString).toString(), ''].join('\n');
  156. // 步骤四:计算 Signature
  157. var qSignature = CryptoJS.HmacSHA1(stringToSign, signKey).toString();
  158. // 步骤五:构造 Authorization
  159. var authorization = ['q-sign-algorithm=' + qSignAlgorithm, 'q-ak=' + qAk, 'q-sign-time=' + qSignTime, 'q-key-time=' + qKeyTime, 'q-header-list=' + qHeaderList, 'q-url-param-list=' + qUrlParamList, 'q-signature=' + qSignature].join('&');
  160. return authorization;
  161. };
  162. var readIntBE = function (chunk, size, offset) {
  163. var bytes = size / 8;
  164. var buf = chunk.slice(offset, offset + bytes);
  165. new Uint8Array(buf).reverse();
  166. return new { 8: Uint8Array, 16: Uint16Array, 32: Uint32Array }[size](buf)[0];
  167. };
  168. var buf2str = function (chunk, start, end, isUtf8) {
  169. var buf = chunk.slice(start, end);
  170. var str = '';
  171. new Uint8Array(buf).forEach(function (charCode) {
  172. str += String.fromCharCode(charCode);
  173. });
  174. if (isUtf8) str = decodeURIComponent(escape(str));
  175. return str;
  176. };
  177. var parseSelectPayload = function (chunk) {
  178. var header = {};
  179. var body = buf2str(chunk);
  180. var result = { records: [] };
  181. while (chunk.byteLength) {
  182. var totalLength = readIntBE(chunk, 32, 0);
  183. var headerLength = readIntBE(chunk, 32, 4);
  184. var payloadRestLength = totalLength - headerLength - 16;
  185. var offset = 0;
  186. var content;
  187. chunk = chunk.slice(12);
  188. // 获取 Message 的 header 信息
  189. while (offset < headerLength) {
  190. var headerNameLength = readIntBE(chunk, 8, offset);
  191. var headerName = buf2str(chunk, offset + 1, offset + 1 + headerNameLength);
  192. var headerValueLength = readIntBE(chunk, 16, offset + headerNameLength + 2);
  193. var headerValue = buf2str(chunk, offset + headerNameLength + 4, offset + headerNameLength + 4 + headerValueLength);
  194. header[headerName] = headerValue;
  195. offset += headerNameLength + 4 + headerValueLength;
  196. }
  197. if (header[':event-type'] === 'Records') {
  198. content = buf2str(chunk, offset, offset + payloadRestLength, true);
  199. result.records.push(content);
  200. } else if (header[':event-type'] === 'Stats') {
  201. content = buf2str(chunk, offset, offset + payloadRestLength, true);
  202. result.stats = util.xml2json(content).Stats;
  203. } else if (header[':event-type'] === 'error') {
  204. var errCode = header[':error-code'];
  205. var errMessage = header[':error-message'];
  206. var err = new Error(errMessage);
  207. err.message = errMessage;
  208. err.name = err.code = errCode;
  209. result.error = err;
  210. } else if (['Progress', 'Continuation', 'End'].includes(header[':event-type'])) {
  211. // do nothing
  212. }
  213. chunk = chunk.slice(offset + payloadRestLength + 4);
  214. }
  215. return {
  216. payload: result.records.join(''),
  217. body: body
  218. };
  219. };
  220. var noop = function () {};
  221. // 清除对象里值为的 undefined 或 null 的属性
  222. var clearKey = function (obj) {
  223. var retObj = {};
  224. for (var key in obj) {
  225. if (obj.hasOwnProperty(key) && obj[key] !== undefined && obj[key] !== null) {
  226. retObj[key] = obj[key];
  227. }
  228. }
  229. return retObj;
  230. };
  231. var readAsBinaryString = function (blob, callback) {
  232. var readFun;
  233. var fr = new FileReader();
  234. if (FileReader.prototype.readAsBinaryString) {
  235. readFun = FileReader.prototype.readAsBinaryString;
  236. fr.onload = function () {
  237. callback(this.result);
  238. };
  239. } else if (FileReader.prototype.readAsArrayBuffer) {
  240. // 在 ie11 添加 readAsBinaryString 兼容
  241. readFun = function (fileData) {
  242. var binary = "";
  243. var pt = this;
  244. var reader = new FileReader();
  245. reader.onload = function (e) {
  246. var bytes = new Uint8Array(reader.result);
  247. var length = bytes.byteLength;
  248. for (var i = 0; i < length; i++) {
  249. binary += String.fromCharCode(bytes[i]);
  250. }
  251. callback(binary);
  252. };
  253. reader.readAsArrayBuffer(fileData);
  254. };
  255. } else {
  256. console.error('FileReader not support readAsBinaryString');
  257. }
  258. readFun.call(fr, blob);
  259. };
  260. var fileSliceNeedCopy = function () {
  261. var compareVersion = function (a, b) {
  262. a = a.split('.');
  263. b = b.split('.');
  264. for (var i = 0; i < b.length; i++) {
  265. if (a[i] !== b[i]) {
  266. return parseInt(a[i]) > parseInt(b[i]) ? 1 : -1;
  267. }
  268. }
  269. return 0;
  270. };
  271. var check = function (ua) {
  272. var ChromeVersion = (ua.match(/Chrome\/([.\d]+)/) || [])[1];
  273. var QBCoreVersion = (ua.match(/QBCore\/([.\d]+)/) || [])[1];
  274. var QQBrowserVersion = (ua.match(/QQBrowser\/([.\d]+)/) || [])[1];
  275. var need = ChromeVersion && compareVersion(ChromeVersion, '53.0.2785.116') < 0 && QBCoreVersion && compareVersion(QBCoreVersion, '3.53.991.400') < 0 && QQBrowserVersion && compareVersion(QQBrowserVersion, '9.0.2524.400') <= 0 || false;
  276. return need;
  277. };
  278. return check(navigator && navigator.userAgent);
  279. }();
  280. // 获取文件分片
  281. var fileSlice = function (file, start, end, isUseToUpload, callback) {
  282. var blob;
  283. if (file.slice) {
  284. blob = file.slice(start, end);
  285. } else if (file.mozSlice) {
  286. blob = file.mozSlice(start, end);
  287. } else if (file.webkitSlice) {
  288. blob = file.webkitSlice(start, end);
  289. }
  290. if (isUseToUpload && fileSliceNeedCopy) {
  291. var reader = new FileReader();
  292. reader.onload = function (e) {
  293. blob = null;
  294. callback(new Blob([reader.result]));
  295. };
  296. reader.readAsArrayBuffer(blob);
  297. } else {
  298. callback(blob);
  299. }
  300. };
  301. // 获取文件内容的 MD5
  302. var getBodyMd5 = function (UploadCheckContentMd5, Body, callback, onProgress) {
  303. callback = callback || noop;
  304. if (UploadCheckContentMd5) {
  305. if (typeof Body === 'string') {
  306. callback(util.md5(Body, true));
  307. } else if (Blob && Body instanceof Blob) {
  308. util.getFileMd5(Body, function (err, md5) {
  309. callback(md5);
  310. }, onProgress);
  311. } else {
  312. callback();
  313. }
  314. } else {
  315. callback();
  316. }
  317. };
  318. // 获取文件 md5 值
  319. var md5ChunkSize = 1024 * 1024;
  320. var getFileMd5 = function (blob, callback, onProgress) {
  321. var size = blob.size;
  322. var loaded = 0;
  323. var md5ctx = md5.getCtx();
  324. var next = function (start) {
  325. if (start >= size) {
  326. var hash = md5ctx.digest('hex');
  327. callback(null, hash);
  328. return;
  329. }
  330. var end = Math.min(size, start + md5ChunkSize);
  331. util.fileSlice(blob, start, end, false, function (chunk) {
  332. readAsBinaryString(chunk, function (content) {
  333. chunk = null;
  334. md5ctx = md5ctx.update(content, true);
  335. loaded += content.length;
  336. content = null;
  337. if (onProgress) onProgress({ loaded: loaded, total: size, percent: Math.round(loaded / size * 10000) / 10000 });
  338. next(start + md5ChunkSize);
  339. });
  340. });
  341. };
  342. next(0);
  343. };
  344. function clone(obj) {
  345. return map(obj, function (v) {
  346. return typeof v === 'object' && v !== null ? clone(v) : v;
  347. });
  348. }
  349. function attr(obj, name, defaultValue) {
  350. return obj && name in obj ? obj[name] : defaultValue;
  351. }
  352. function extend(target, source) {
  353. each(source, function (val, key) {
  354. target[key] = source[key];
  355. });
  356. return target;
  357. }
  358. function isArray(arr) {
  359. return arr instanceof Array;
  360. }
  361. function isInArray(arr, item) {
  362. var flag = false;
  363. for (var i = 0; i < arr.length; i++) {
  364. if (item === arr[i]) {
  365. flag = true;
  366. break;
  367. }
  368. }
  369. return flag;
  370. }
  371. function makeArray(arr) {
  372. return isArray(arr) ? arr : [arr];
  373. }
  374. function each(obj, fn) {
  375. for (var i in obj) {
  376. if (obj.hasOwnProperty(i)) {
  377. fn(obj[i], i);
  378. }
  379. }
  380. }
  381. function map(obj, fn) {
  382. var o = isArray(obj) ? [] : {};
  383. for (var i in obj) {
  384. if (obj.hasOwnProperty(i)) {
  385. o[i] = fn(obj[i], i);
  386. }
  387. }
  388. return o;
  389. }
  390. function filter(obj, fn) {
  391. var iaArr = isArray(obj);
  392. var o = iaArr ? [] : {};
  393. for (var i in obj) {
  394. if (obj.hasOwnProperty(i)) {
  395. if (fn(obj[i], i)) {
  396. if (iaArr) {
  397. o.push(obj[i]);
  398. } else {
  399. o[i] = obj[i];
  400. }
  401. }
  402. }
  403. }
  404. return o;
  405. }
  406. var binaryBase64 = function (str) {
  407. var i,
  408. len,
  409. char,
  410. res = '';
  411. for (i = 0, len = str.length / 2; i < len; i++) {
  412. char = parseInt(str[i * 2] + str[i * 2 + 1], 16);
  413. res += String.fromCharCode(char);
  414. }
  415. return btoa(res);
  416. };
  417. var uuid = function () {
  418. var S4 = function () {
  419. return ((1 + Math.random()) * 0x10000 | 0).toString(16).substring(1);
  420. };
  421. return S4() + S4() + "-" + S4() + "-" + S4() + "-" + S4() + "-" + S4() + S4() + S4();
  422. };
  423. var hasMissingParams = function (apiName, params) {
  424. var Bucket = params.Bucket;
  425. var Region = params.Region;
  426. var Key = params.Key;
  427. if (apiName.indexOf('Bucket') > -1 || apiName === 'deleteMultipleObject' || apiName === 'multipartList' || apiName === 'listObjectVersions') {
  428. if (!Bucket) return 'Bucket';
  429. if (!Region) return 'Region';
  430. } else if (apiName.indexOf('Object') > -1 || apiName.indexOf('multipart') > -1 || apiName === 'sliceUploadFile' || apiName === 'abortUploadTask') {
  431. if (!Bucket) return 'Bucket';
  432. if (!Region) return 'Region';
  433. if (!Key) return 'Key';
  434. }
  435. return false;
  436. };
  437. var formatParams = function (apiName, params) {
  438. // 复制参数对象
  439. params = extend({}, params);
  440. // 统一处理 Headers
  441. if (apiName !== 'getAuth' && apiName !== 'getV4Auth' && apiName !== 'getObjectUrl') {
  442. var Headers = params.Headers || {};
  443. if (params && typeof params === 'object') {
  444. (function () {
  445. for (var key in params) {
  446. if (params.hasOwnProperty(key) && key.indexOf('x-cos-') > -1) {
  447. Headers[key] = params[key];
  448. }
  449. }
  450. })();
  451. var headerMap = {
  452. // params headers
  453. 'x-cos-mfa': 'MFA',
  454. 'Content-MD5': 'ContentMD5',
  455. 'Content-Length': 'ContentLength',
  456. 'Content-Type': 'ContentType',
  457. 'Expect': 'Expect',
  458. 'Expires': 'Expires',
  459. 'Cache-Control': 'CacheControl',
  460. 'Content-Disposition': 'ContentDisposition',
  461. 'Content-Encoding': 'ContentEncoding',
  462. 'Range': 'Range',
  463. 'If-Modified-Since': 'IfModifiedSince',
  464. 'If-Unmodified-Since': 'IfUnmodifiedSince',
  465. 'If-Match': 'IfMatch',
  466. 'If-None-Match': 'IfNoneMatch',
  467. 'x-cos-copy-source': 'CopySource',
  468. 'x-cos-copy-source-Range': 'CopySourceRange',
  469. 'x-cos-metadata-directive': 'MetadataDirective',
  470. 'x-cos-copy-source-If-Modified-Since': 'CopySourceIfModifiedSince',
  471. 'x-cos-copy-source-If-Unmodified-Since': 'CopySourceIfUnmodifiedSince',
  472. 'x-cos-copy-source-If-Match': 'CopySourceIfMatch',
  473. 'x-cos-copy-source-If-None-Match': 'CopySourceIfNoneMatch',
  474. 'x-cos-acl': 'ACL',
  475. 'x-cos-grant-read': 'GrantRead',
  476. 'x-cos-grant-write': 'GrantWrite',
  477. 'x-cos-grant-full-control': 'GrantFullControl',
  478. 'x-cos-grant-read-acp': 'GrantReadAcp',
  479. 'x-cos-grant-write-acp': 'GrantWriteAcp',
  480. 'x-cos-storage-class': 'StorageClass',
  481. 'x-cos-traffic-limit': 'TrafficLimit',
  482. // SSE-C
  483. 'x-cos-server-side-encryption-customer-algorithm': 'SSECustomerAlgorithm',
  484. 'x-cos-server-side-encryption-customer-key': 'SSECustomerKey',
  485. 'x-cos-server-side-encryption-customer-key-MD5': 'SSECustomerKeyMD5',
  486. // SSE-COS、SSE-KMS
  487. 'x-cos-server-side-encryption': 'ServerSideEncryption',
  488. 'x-cos-server-side-encryption-cos-kms-key-id': 'SSEKMSKeyId',
  489. 'x-cos-server-side-encryption-context': 'SSEContext'
  490. };
  491. util.each(headerMap, function (paramKey, headerKey) {
  492. if (params[paramKey] !== undefined) {
  493. Headers[headerKey] = params[paramKey];
  494. }
  495. });
  496. params.Headers = clearKey(Headers);
  497. }
  498. }
  499. return params;
  500. };
  501. var apiWrapper = function (apiName, apiFn) {
  502. return function (params, callback) {
  503. var self = this;
  504. // 处理参数
  505. if (typeof params === 'function') {
  506. callback = params;
  507. params = {};
  508. }
  509. // 整理参数格式
  510. params = formatParams(apiName, params);
  511. // 代理回调函数
  512. var formatResult = function (result) {
  513. if (result && result.headers) {
  514. result.headers['x-cos-request-id'] && (result.RequestId = result.headers['x-cos-request-id']);
  515. result.headers['x-cos-version-id'] && (result.VersionId = result.headers['x-cos-version-id']);
  516. result.headers['x-cos-delete-marker'] && (result.DeleteMarker = result.headers['x-cos-delete-marker']);
  517. }
  518. return result;
  519. };
  520. var _callback = function (err, data) {
  521. callback && callback(formatResult(err), formatResult(data));
  522. };
  523. var checkParams = function () {
  524. if (apiName !== 'getService' && apiName !== 'abortUploadTask') {
  525. // 判断参数是否完整
  526. var missingResult = hasMissingParams(apiName, params);
  527. if (missingResult) {
  528. return 'missing param ' + missingResult;
  529. }
  530. // 判断 region 格式
  531. if (params.Region) {
  532. if (params.Region.indexOf('cos.') > -1) {
  533. return 'param Region should not be start with "cos."';
  534. } else if (!/^([a-z\d-]+)$/.test(params.Region)) {
  535. return 'Region format error.';
  536. }
  537. // 判断 region 格式
  538. if (!self.options.CompatibilityMode && params.Region.indexOf('-') === -1 && params.Region !== 'yfb' && params.Region !== 'default' && params.Region !== 'accelerate') {
  539. console.warn('warning: param Region format error, find help here: https://cloud.tencent.com/document/product/436/6224');
  540. }
  541. }
  542. // 兼容不带 AppId 的 Bucket
  543. if (params.Bucket) {
  544. if (!/^([a-z\d-]+)-(\d+)$/.test(params.Bucket)) {
  545. if (params.AppId) {
  546. params.Bucket = params.Bucket + '-' + params.AppId;
  547. } else if (self.options.AppId) {
  548. params.Bucket = params.Bucket + '-' + self.options.AppId;
  549. } else {
  550. return 'Bucket should format as "test-1250000000".';
  551. }
  552. }
  553. if (params.AppId) {
  554. console.warn('warning: AppId has been deprecated, Please put it at the end of parameter Bucket(E.g Bucket:"test-1250000000" ).');
  555. delete params.AppId;
  556. }
  557. }
  558. // 如果 Key 是 / 开头,强制去掉第一个 /
  559. if (!self.options.UseRawKey && params.Key && params.Key.substr(0, 1) === '/') {
  560. params.Key = params.Key.substr(1);
  561. }
  562. }
  563. };
  564. var errMsg = checkParams();
  565. var isSync = apiName === 'getAuth' || apiName === 'getObjectUrl';
  566. if (window.Promise && !isSync && !callback) {
  567. return new Promise(function (resolve, reject) {
  568. callback = function (err, data) {
  569. err ? reject(err) : resolve(data);
  570. };
  571. if (errMsg) return _callback(util.error(new Error(errMsg)));
  572. apiFn.call(self, params, _callback);
  573. });
  574. } else {
  575. if (errMsg) return _callback(util.error(new Error(errMsg)));
  576. var res = apiFn.call(self, params, _callback);
  577. if (isSync) return res;
  578. }
  579. };
  580. };
  581. var throttleOnProgress = function (total, onProgress) {
  582. var self = this;
  583. var size0 = 0;
  584. var size1 = 0;
  585. var time0 = Date.now();
  586. var time1;
  587. var timer;
  588. function update() {
  589. timer = 0;
  590. if (onProgress && typeof onProgress === 'function') {
  591. time1 = Date.now();
  592. var speed = Math.max(0, Math.round((size1 - size0) / ((time1 - time0) / 1000) * 100) / 100) || 0;
  593. var percent;
  594. if (size1 === 0 && total === 0) {
  595. percent = 1;
  596. } else {
  597. percent = Math.floor(size1 / total * 100) / 100 || 0;
  598. }
  599. time0 = time1;
  600. size0 = size1;
  601. try {
  602. onProgress({ loaded: size1, total: total, speed: speed, percent: percent });
  603. } catch (e) {}
  604. }
  605. }
  606. return function (info, immediately) {
  607. if (info) {
  608. size1 = info.loaded;
  609. total = info.total;
  610. }
  611. if (immediately) {
  612. clearTimeout(timer);
  613. update();
  614. } else {
  615. if (timer) return;
  616. timer = setTimeout(update, self.options.ProgressInterval);
  617. }
  618. };
  619. };
  620. var getFileSize = function (api, params, callback) {
  621. var size;
  622. if (typeof params.Body === 'string') {
  623. params.Body = new Blob([params.Body], { type: 'text/plain' });
  624. } else if (params.Body instanceof ArrayBuffer) {
  625. params.Body = new Blob([params.Body]);
  626. }
  627. if (params.Body && (params.Body instanceof Blob || params.Body.toString() === '[object File]' || params.Body.toString() === '[object Blob]')) {
  628. size = params.Body.size;
  629. } else {
  630. callback(util.error(new Error('params body format error, Only allow File|Blob|String.')));
  631. return;
  632. }
  633. params.ContentLength = size;
  634. callback(null, size);
  635. };
  636. // 获取调正的时间戳
  637. var getSkewTime = function (offset) {
  638. return Date.now() + (offset || 0);
  639. };
  640. var error = function (err, opt) {
  641. var sourceErr = err;
  642. err.message = err.message || null;
  643. if (typeof opt === 'string') {
  644. err.error = opt;
  645. err.message = opt;
  646. } else if (typeof opt === 'object' && opt !== null) {
  647. extend(err, opt);
  648. if (opt.code || opt.name) err.code = opt.code || opt.name;
  649. if (opt.message) err.message = opt.message;
  650. if (opt.stack) err.stack = opt.stack;
  651. }
  652. if (typeof Object.defineProperty === 'function') {
  653. Object.defineProperty(err, 'name', { writable: true, enumerable: false });
  654. Object.defineProperty(err, 'message', { enumerable: true });
  655. }
  656. err.name = opt && opt.name || err.name || err.code || 'Error';
  657. if (!err.code) err.code = err.name;
  658. if (!err.error) err.error = clone(sourceErr); // 兼容老的错误格式
  659. return err;
  660. };
  661. var isNode = function () {
  662. return typeof window !== 'object' && typeof process === 'object' && "function" === 'function';
  663. };
  664. var util = {
  665. noop: noop,
  666. formatParams: formatParams,
  667. apiWrapper: apiWrapper,
  668. xml2json: xml2json,
  669. json2xml: json2xml,
  670. md5: md5,
  671. clearKey: clearKey,
  672. fileSlice: fileSlice,
  673. getBodyMd5: getBodyMd5,
  674. getFileMd5: getFileMd5,
  675. binaryBase64: binaryBase64,
  676. extend: extend,
  677. isArray: isArray,
  678. isInArray: isInArray,
  679. makeArray: makeArray,
  680. each: each,
  681. map: map,
  682. filter: filter,
  683. clone: clone,
  684. attr: attr,
  685. uuid: uuid,
  686. camSafeUrlEncode: camSafeUrlEncode,
  687. throttleOnProgress: throttleOnProgress,
  688. getFileSize: getFileSize,
  689. getSkewTime: getSkewTime,
  690. error: error,
  691. getAuth: getAuth,
  692. parseSelectPayload: parseSelectPayload,
  693. isBrowser: true,
  694. isNode: isNode
  695. };
  696. module.exports = util;
  697. /* WEBPACK VAR INJECTION */}.call(exports, __webpack_require__(1)))
  698. /***/ }),
  699. /* 1 */
  700. /***/ (function(module, exports) {
  701. // shim for using process in browser
  702. var process = module.exports = {};
  703. // cached from whatever global is present so that test runners that stub it
  704. // don't break things. But we need to wrap it in a try catch in case it is
  705. // wrapped in strict mode code which doesn't define any globals. It's inside a
  706. // function because try/catches deoptimize in certain engines.
  707. var cachedSetTimeout;
  708. var cachedClearTimeout;
  709. function defaultSetTimout() {
  710. throw new Error('setTimeout has not been defined');
  711. }
  712. function defaultClearTimeout () {
  713. throw new Error('clearTimeout has not been defined');
  714. }
  715. (function () {
  716. try {
  717. if (typeof setTimeout === 'function') {
  718. cachedSetTimeout = setTimeout;
  719. } else {
  720. cachedSetTimeout = defaultSetTimout;
  721. }
  722. } catch (e) {
  723. cachedSetTimeout = defaultSetTimout;
  724. }
  725. try {
  726. if (typeof clearTimeout === 'function') {
  727. cachedClearTimeout = clearTimeout;
  728. } else {
  729. cachedClearTimeout = defaultClearTimeout;
  730. }
  731. } catch (e) {
  732. cachedClearTimeout = defaultClearTimeout;
  733. }
  734. } ())
  735. function runTimeout(fun) {
  736. if (cachedSetTimeout === setTimeout) {
  737. //normal enviroments in sane situations
  738. return setTimeout(fun, 0);
  739. }
  740. // if setTimeout wasn't available but was latter defined
  741. if ((cachedSetTimeout === defaultSetTimout || !cachedSetTimeout) && setTimeout) {
  742. cachedSetTimeout = setTimeout;
  743. return setTimeout(fun, 0);
  744. }
  745. try {
  746. // when when somebody has screwed with setTimeout but no I.E. maddness
  747. return cachedSetTimeout(fun, 0);
  748. } catch(e){
  749. try {
  750. // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally
  751. return cachedSetTimeout.call(null, fun, 0);
  752. } catch(e){
  753. // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error
  754. return cachedSetTimeout.call(this, fun, 0);
  755. }
  756. }
  757. }
  758. function runClearTimeout(marker) {
  759. if (cachedClearTimeout === clearTimeout) {
  760. //normal enviroments in sane situations
  761. return clearTimeout(marker);
  762. }
  763. // if clearTimeout wasn't available but was latter defined
  764. if ((cachedClearTimeout === defaultClearTimeout || !cachedClearTimeout) && clearTimeout) {
  765. cachedClearTimeout = clearTimeout;
  766. return clearTimeout(marker);
  767. }
  768. try {
  769. // when when somebody has screwed with setTimeout but no I.E. maddness
  770. return cachedClearTimeout(marker);
  771. } catch (e){
  772. try {
  773. // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally
  774. return cachedClearTimeout.call(null, marker);
  775. } catch (e){
  776. // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error.
  777. // Some versions of I.E. have different rules for clearTimeout vs setTimeout
  778. return cachedClearTimeout.call(this, marker);
  779. }
  780. }
  781. }
  782. var queue = [];
  783. var draining = false;
  784. var currentQueue;
  785. var queueIndex = -1;
  786. function cleanUpNextTick() {
  787. if (!draining || !currentQueue) {
  788. return;
  789. }
  790. draining = false;
  791. if (currentQueue.length) {
  792. queue = currentQueue.concat(queue);
  793. } else {
  794. queueIndex = -1;
  795. }
  796. if (queue.length) {
  797. drainQueue();
  798. }
  799. }
  800. function drainQueue() {
  801. if (draining) {
  802. return;
  803. }
  804. var timeout = runTimeout(cleanUpNextTick);
  805. draining = true;
  806. var len = queue.length;
  807. while(len) {
  808. currentQueue = queue;
  809. queue = [];
  810. while (++queueIndex < len) {
  811. if (currentQueue) {
  812. currentQueue[queueIndex].run();
  813. }
  814. }
  815. queueIndex = -1;
  816. len = queue.length;
  817. }
  818. currentQueue = null;
  819. draining = false;
  820. runClearTimeout(timeout);
  821. }
  822. process.nextTick = function (fun) {
  823. var args = new Array(arguments.length - 1);
  824. if (arguments.length > 1) {
  825. for (var i = 1; i < arguments.length; i++) {
  826. args[i - 1] = arguments[i];
  827. }
  828. }
  829. queue.push(new Item(fun, args));
  830. if (queue.length === 1 && !draining) {
  831. runTimeout(drainQueue);
  832. }
  833. };
  834. // v8 likes predictible objects
  835. function Item(fun, array) {
  836. this.fun = fun;
  837. this.array = array;
  838. }
  839. Item.prototype.run = function () {
  840. this.fun.apply(null, this.array);
  841. };
  842. process.title = 'browser';
  843. process.browser = true;
  844. process.env = {};
  845. process.argv = [];
  846. process.version = ''; // empty string to avoid regexp issues
  847. process.versions = {};
  848. function noop() {}
  849. process.on = noop;
  850. process.addListener = noop;
  851. process.once = noop;
  852. process.off = noop;
  853. process.removeListener = noop;
  854. process.removeAllListeners = noop;
  855. process.emit = noop;
  856. process.prependListener = noop;
  857. process.prependOnceListener = noop;
  858. process.listeners = function (name) { return [] }
  859. process.binding = function (name) {
  860. throw new Error('process.binding is not supported');
  861. };
  862. process.cwd = function () { return '/' };
  863. process.chdir = function (dir) {
  864. throw new Error('process.chdir is not supported');
  865. };
  866. process.umask = function() { return 0; };
  867. /***/ }),
  868. /* 2 */
  869. /***/ (function(module, exports) {
  870. /*
  871. * DOM Level 2
  872. * Object DOMException
  873. * @see http://www.w3.org/TR/REC-DOM-Level-1/ecma-script-language-binding.html
  874. * @see http://www.w3.org/TR/2000/REC-DOM-Level-2-Core-20001113/ecma-script-binding.html
  875. */
  876. function copy(src,dest){
  877. for(var p in src){
  878. dest[p] = src[p];
  879. }
  880. }
  881. /**
  882. ^\w+\.prototype\.([_\w]+)\s*=\s*((?:.*\{\s*?[\r\n][\s\S]*?^})|\S.*?(?=[;\r\n]));?
  883. ^\w+\.prototype\.([_\w]+)\s*=\s*(\S.*?(?=[;\r\n]));?
  884. */
  885. function _extends(Class,Super){
  886. var pt = Class.prototype;
  887. if(Object.create){
  888. var ppt = Object.create(Super.prototype)
  889. pt.__proto__ = ppt;
  890. }
  891. if(!(pt instanceof Super)){
  892. function t(){};
  893. t.prototype = Super.prototype;
  894. t = new t();
  895. copy(pt,t);
  896. Class.prototype = pt = t;
  897. }
  898. if(pt.constructor != Class){
  899. if(typeof Class != 'function'){
  900. console.error("unknow Class:"+Class)
  901. }
  902. pt.constructor = Class
  903. }
  904. }
  905. var htmlns = 'http://www.w3.org/1999/xhtml' ;
  906. // Node Types
  907. var NodeType = {}
  908. var ELEMENT_NODE = NodeType.ELEMENT_NODE = 1;
  909. var ATTRIBUTE_NODE = NodeType.ATTRIBUTE_NODE = 2;
  910. var TEXT_NODE = NodeType.TEXT_NODE = 3;
  911. var CDATA_SECTION_NODE = NodeType.CDATA_SECTION_NODE = 4;
  912. var ENTITY_REFERENCE_NODE = NodeType.ENTITY_REFERENCE_NODE = 5;
  913. var ENTITY_NODE = NodeType.ENTITY_NODE = 6;
  914. var PROCESSING_INSTRUCTION_NODE = NodeType.PROCESSING_INSTRUCTION_NODE = 7;
  915. var COMMENT_NODE = NodeType.COMMENT_NODE = 8;
  916. var DOCUMENT_NODE = NodeType.DOCUMENT_NODE = 9;
  917. var DOCUMENT_TYPE_NODE = NodeType.DOCUMENT_TYPE_NODE = 10;
  918. var DOCUMENT_FRAGMENT_NODE = NodeType.DOCUMENT_FRAGMENT_NODE = 11;
  919. var NOTATION_NODE = NodeType.NOTATION_NODE = 12;
  920. // ExceptionCode
  921. var ExceptionCode = {}
  922. var ExceptionMessage = {};
  923. var INDEX_SIZE_ERR = ExceptionCode.INDEX_SIZE_ERR = ((ExceptionMessage[1]="Index size error"),1);
  924. var DOMSTRING_SIZE_ERR = ExceptionCode.DOMSTRING_SIZE_ERR = ((ExceptionMessage[2]="DOMString size error"),2);
  925. var HIERARCHY_REQUEST_ERR = ExceptionCode.HIERARCHY_REQUEST_ERR = ((ExceptionMessage[3]="Hierarchy request error"),3);
  926. var WRONG_DOCUMENT_ERR = ExceptionCode.WRONG_DOCUMENT_ERR = ((ExceptionMessage[4]="Wrong document"),4);
  927. var INVALID_CHARACTER_ERR = ExceptionCode.INVALID_CHARACTER_ERR = ((ExceptionMessage[5]="Invalid character"),5);
  928. var NO_DATA_ALLOWED_ERR = ExceptionCode.NO_DATA_ALLOWED_ERR = ((ExceptionMessage[6]="No data allowed"),6);
  929. var NO_MODIFICATION_ALLOWED_ERR = ExceptionCode.NO_MODIFICATION_ALLOWED_ERR = ((ExceptionMessage[7]="No modification allowed"),7);
  930. var NOT_FOUND_ERR = ExceptionCode.NOT_FOUND_ERR = ((ExceptionMessage[8]="Not found"),8);
  931. var NOT_SUPPORTED_ERR = ExceptionCode.NOT_SUPPORTED_ERR = ((ExceptionMessage[9]="Not supported"),9);
  932. var INUSE_ATTRIBUTE_ERR = ExceptionCode.INUSE_ATTRIBUTE_ERR = ((ExceptionMessage[10]="Attribute in use"),10);
  933. //level2
  934. var INVALID_STATE_ERR = ExceptionCode.INVALID_STATE_ERR = ((ExceptionMessage[11]="Invalid state"),11);
  935. var SYNTAX_ERR = ExceptionCode.SYNTAX_ERR = ((ExceptionMessage[12]="Syntax error"),12);
  936. var INVALID_MODIFICATION_ERR = ExceptionCode.INVALID_MODIFICATION_ERR = ((ExceptionMessage[13]="Invalid modification"),13);
  937. var NAMESPACE_ERR = ExceptionCode.NAMESPACE_ERR = ((ExceptionMessage[14]="Invalid namespace"),14);
  938. var INVALID_ACCESS_ERR = ExceptionCode.INVALID_ACCESS_ERR = ((ExceptionMessage[15]="Invalid access"),15);
  939. function DOMException(code, message) {
  940. if(message instanceof Error){
  941. var error = message;
  942. }else{
  943. error = this;
  944. Error.call(this, ExceptionMessage[code]);
  945. this.message = ExceptionMessage[code];
  946. if(Error.captureStackTrace) Error.captureStackTrace(this, DOMException);
  947. }
  948. error.code = code;
  949. if(message) this.message = this.message + ": " + message;
  950. return error;
  951. };
  952. DOMException.prototype = Error.prototype;
  953. copy(ExceptionCode,DOMException)
  954. /**
  955. * @see http://www.w3.org/TR/2000/REC-DOM-Level-2-Core-20001113/core.html#ID-536297177
  956. * The NodeList interface provides the abstraction of an ordered collection of nodes, without defining or constraining how this collection is implemented. NodeList objects in the DOM are live.
  957. * The items in the NodeList are accessible via an integral index, starting from 0.
  958. */
  959. function NodeList() {
  960. };
  961. NodeList.prototype = {
  962. /**
  963. * The number of nodes in the list. The range of valid child node indices is 0 to length-1 inclusive.
  964. * @standard level1
  965. */
  966. length:0,
  967. /**
  968. * Returns the indexth item in the collection. If index is greater than or equal to the number of nodes in the list, this returns null.
  969. * @standard level1
  970. * @param index unsigned long
  971. * Index into the collection.
  972. * @return Node
  973. * The node at the indexth position in the NodeList, or null if that is not a valid index.
  974. */
  975. item: function(index) {
  976. return this[index] || null;
  977. },
  978. toString:function(isHTML,nodeFilter){
  979. for(var buf = [], i = 0;i<this.length;i++){
  980. serializeToString(this[i],buf,isHTML,nodeFilter);
  981. }
  982. return buf.join('');
  983. }
  984. };
  985. function LiveNodeList(node,refresh){
  986. this._node = node;
  987. this._refresh = refresh
  988. _updateLiveList(this);
  989. }
  990. function _updateLiveList(list){
  991. var inc = list._node._inc || list._node.ownerDocument._inc;
  992. if(list._inc != inc){
  993. var ls = list._refresh(list._node);
  994. //console.log(ls.length)
  995. __set__(list,'length',ls.length);
  996. copy(ls,list);
  997. list._inc = inc;
  998. }
  999. }
  1000. LiveNodeList.prototype.item = function(i){
  1001. _updateLiveList(this);
  1002. return this[i];
  1003. }
  1004. _extends(LiveNodeList,NodeList);
  1005. /**
  1006. *
  1007. * Objects implementing the NamedNodeMap interface are used to represent collections of nodes that can be accessed by name. Note that NamedNodeMap does not inherit from NodeList; NamedNodeMaps are not maintained in any particular order. Objects contained in an object implementing NamedNodeMap may also be accessed by an ordinal index, but this is simply to allow convenient enumeration of the contents of a NamedNodeMap, and does not imply that the DOM specifies an order to these Nodes.
  1008. * NamedNodeMap objects in the DOM are live.
  1009. * used for attributes or DocumentType entities
  1010. */
  1011. function NamedNodeMap() {
  1012. };
  1013. function _findNodeIndex(list,node){
  1014. var i = list.length;
  1015. while(i--){
  1016. if(list[i] === node){return i}
  1017. }
  1018. }
  1019. function _addNamedNode(el,list,newAttr,oldAttr){
  1020. if(oldAttr){
  1021. list[_findNodeIndex(list,oldAttr)] = newAttr;
  1022. }else{
  1023. list[list.length++] = newAttr;
  1024. }
  1025. if(el){
  1026. newAttr.ownerElement = el;
  1027. var doc = el.ownerDocument;
  1028. if(doc){
  1029. oldAttr && _onRemoveAttribute(doc,el,oldAttr);
  1030. _onAddAttribute(doc,el,newAttr);
  1031. }
  1032. }
  1033. }
  1034. function _removeNamedNode(el,list,attr){
  1035. //console.log('remove attr:'+attr)
  1036. var i = _findNodeIndex(list,attr);
  1037. if(i>=0){
  1038. var lastIndex = list.length-1
  1039. while(i<lastIndex){
  1040. list[i] = list[++i]
  1041. }
  1042. list.length = lastIndex;
  1043. if(el){
  1044. var doc = el.ownerDocument;
  1045. if(doc){
  1046. _onRemoveAttribute(doc,el,attr);
  1047. attr.ownerElement = null;
  1048. }
  1049. }
  1050. }else{
  1051. throw DOMException(NOT_FOUND_ERR,new Error(el.tagName+'@'+attr))
  1052. }
  1053. }
  1054. NamedNodeMap.prototype = {
  1055. length:0,
  1056. item:NodeList.prototype.item,
  1057. getNamedItem: function(key) {
  1058. // if(key.indexOf(':')>0 || key == 'xmlns'){
  1059. // return null;
  1060. // }
  1061. //console.log()
  1062. var i = this.length;
  1063. while(i--){
  1064. var attr = this[i];
  1065. //console.log(attr.nodeName,key)
  1066. if(attr.nodeName == key){
  1067. return attr;
  1068. }
  1069. }
  1070. },
  1071. setNamedItem: function(attr) {
  1072. var el = attr.ownerElement;
  1073. if(el && el!=this._ownerElement){
  1074. throw new DOMException(INUSE_ATTRIBUTE_ERR);
  1075. }
  1076. var oldAttr = this.getNamedItem(attr.nodeName);
  1077. _addNamedNode(this._ownerElement,this,attr,oldAttr);
  1078. return oldAttr;
  1079. },
  1080. /* returns Node */
  1081. setNamedItemNS: function(attr) {// raises: WRONG_DOCUMENT_ERR,NO_MODIFICATION_ALLOWED_ERR,INUSE_ATTRIBUTE_ERR
  1082. var el = attr.ownerElement, oldAttr;
  1083. if(el && el!=this._ownerElement){
  1084. throw new DOMException(INUSE_ATTRIBUTE_ERR);
  1085. }
  1086. oldAttr = this.getNamedItemNS(attr.namespaceURI,attr.localName);
  1087. _addNamedNode(this._ownerElement,this,attr,oldAttr);
  1088. return oldAttr;
  1089. },
  1090. /* returns Node */
  1091. removeNamedItem: function(key) {
  1092. var attr = this.getNamedItem(key);
  1093. _removeNamedNode(this._ownerElement,this,attr);
  1094. return attr;
  1095. },// raises: NOT_FOUND_ERR,NO_MODIFICATION_ALLOWED_ERR
  1096. //for level2
  1097. removeNamedItemNS:function(namespaceURI,localName){
  1098. var attr = this.getNamedItemNS(namespaceURI,localName);
  1099. _removeNamedNode(this._ownerElement,this,attr);
  1100. return attr;
  1101. },
  1102. getNamedItemNS: function(namespaceURI, localName) {
  1103. var i = this.length;
  1104. while(i--){
  1105. var node = this[i];
  1106. if(node.localName == localName && node.namespaceURI == namespaceURI){
  1107. return node;
  1108. }
  1109. }
  1110. return null;
  1111. }
  1112. };
  1113. /**
  1114. * @see http://www.w3.org/TR/REC-DOM-Level-1/level-one-core.html#ID-102161490
  1115. */
  1116. function DOMImplementation(/* Object */ features) {
  1117. this._features = {};
  1118. if (features) {
  1119. for (var feature in features) {
  1120. this._features = features[feature];
  1121. }
  1122. }
  1123. };
  1124. DOMImplementation.prototype = {
  1125. hasFeature: function(/* string */ feature, /* string */ version) {
  1126. var versions = this._features[feature.toLowerCase()];
  1127. if (versions && (!version || version in versions)) {
  1128. return true;
  1129. } else {
  1130. return false;
  1131. }
  1132. },
  1133. // Introduced in DOM Level 2:
  1134. createDocument:function(namespaceURI, qualifiedName, doctype){// raises:INVALID_CHARACTER_ERR,NAMESPACE_ERR,WRONG_DOCUMENT_ERR
  1135. var doc = new Document();
  1136. doc.implementation = this;
  1137. doc.childNodes = new NodeList();
  1138. doc.doctype = doctype;
  1139. if(doctype){
  1140. doc.appendChild(doctype);
  1141. }
  1142. if(qualifiedName){
  1143. var root = doc.createElementNS(namespaceURI,qualifiedName);
  1144. doc.appendChild(root);
  1145. }
  1146. return doc;
  1147. },
  1148. // Introduced in DOM Level 2:
  1149. createDocumentType:function(qualifiedName, publicId, systemId){// raises:INVALID_CHARACTER_ERR,NAMESPACE_ERR
  1150. var node = new DocumentType();
  1151. node.name = qualifiedName;
  1152. node.nodeName = qualifiedName;
  1153. node.publicId = publicId;
  1154. node.systemId = systemId;
  1155. // Introduced in DOM Level 2:
  1156. //readonly attribute DOMString internalSubset;
  1157. //TODO:..
  1158. // readonly attribute NamedNodeMap entities;
  1159. // readonly attribute NamedNodeMap notations;
  1160. return node;
  1161. }
  1162. };
  1163. /**
  1164. * @see http://www.w3.org/TR/2000/REC-DOM-Level-2-Core-20001113/core.html#ID-1950641247
  1165. */
  1166. function Node() {
  1167. };
  1168. Node.prototype = {
  1169. firstChild : null,
  1170. lastChild : null,
  1171. previousSibling : null,
  1172. nextSibling : null,
  1173. attributes : null,
  1174. parentNode : null,
  1175. childNodes : null,
  1176. ownerDocument : null,
  1177. nodeValue : null,
  1178. namespaceURI : null,
  1179. prefix : null,
  1180. localName : null,
  1181. // Modified in DOM Level 2:
  1182. insertBefore:function(newChild, refChild){//raises
  1183. return _insertBefore(this,newChild,refChild);
  1184. },
  1185. replaceChild:function(newChild, oldChild){//raises
  1186. this.insertBefore(newChild,oldChild);
  1187. if(oldChild){
  1188. this.removeChild(oldChild);
  1189. }
  1190. },
  1191. removeChild:function(oldChild){
  1192. return _removeChild(this,oldChild);
  1193. },
  1194. appendChild:function(newChild){
  1195. return this.insertBefore(newChild,null);
  1196. },
  1197. hasChildNodes:function(){
  1198. return this.firstChild != null;
  1199. },
  1200. cloneNode:function(deep){
  1201. return cloneNode(this.ownerDocument||this,this,deep);
  1202. },
  1203. // Modified in DOM Level 2:
  1204. normalize:function(){
  1205. var child = this.firstChild;
  1206. while(child){
  1207. var next = child.nextSibling;
  1208. if(next && next.nodeType == TEXT_NODE && child.nodeType == TEXT_NODE){
  1209. this.removeChild(next);
  1210. child.appendData(next.data);
  1211. }else{
  1212. child.normalize();
  1213. child = next;
  1214. }
  1215. }
  1216. },
  1217. // Introduced in DOM Level 2:
  1218. isSupported:function(feature, version){
  1219. return this.ownerDocument.implementation.hasFeature(feature,version);
  1220. },
  1221. // Introduced in DOM Level 2:
  1222. hasAttributes:function(){
  1223. return this.attributes.length>0;
  1224. },
  1225. lookupPrefix:function(namespaceURI){
  1226. var el = this;
  1227. while(el){
  1228. var map = el._nsMap;
  1229. //console.dir(map)
  1230. if(map){
  1231. for(var n in map){
  1232. if(map[n] == namespaceURI){
  1233. return n;
  1234. }
  1235. }
  1236. }
  1237. el = el.nodeType == ATTRIBUTE_NODE?el.ownerDocument : el.parentNode;
  1238. }
  1239. return null;
  1240. },
  1241. // Introduced in DOM Level 3:
  1242. lookupNamespaceURI:function(prefix){
  1243. var el = this;
  1244. while(el){
  1245. var map = el._nsMap;
  1246. //console.dir(map)
  1247. if(map){
  1248. if(prefix in map){
  1249. return map[prefix] ;
  1250. }
  1251. }
  1252. el = el.nodeType == ATTRIBUTE_NODE?el.ownerDocument : el.parentNode;
  1253. }
  1254. return null;
  1255. },
  1256. // Introduced in DOM Level 3:
  1257. isDefaultNamespace:function(namespaceURI){
  1258. var prefix = this.lookupPrefix(namespaceURI);
  1259. return prefix == null;
  1260. }
  1261. };
  1262. function _xmlEncoder(c){
  1263. return c == '<' && '&lt;' ||
  1264. c == '>' && '&gt;' ||
  1265. c == '&' && '&amp;' ||
  1266. c == '"' && '&quot;' ||
  1267. '&#'+c.charCodeAt()+';'
  1268. }
  1269. copy(NodeType,Node);
  1270. copy(NodeType,Node.prototype);
  1271. /**
  1272. * @param callback return true for continue,false for break
  1273. * @return boolean true: break visit;
  1274. */
  1275. function _visitNode(node,callback){
  1276. if(callback(node)){
  1277. return true;
  1278. }
  1279. if(node = node.firstChild){
  1280. do{
  1281. if(_visitNode(node,callback)){return true}
  1282. }while(node=node.nextSibling)
  1283. }
  1284. }
  1285. function Document(){
  1286. }
  1287. function _onAddAttribute(doc,el,newAttr){
  1288. doc && doc._inc++;
  1289. var ns = newAttr.namespaceURI ;
  1290. if(ns == 'http://www.w3.org/2000/xmlns/'){
  1291. //update namespace
  1292. el._nsMap[newAttr.prefix?newAttr.localName:''] = newAttr.value
  1293. }
  1294. }
  1295. function _onRemoveAttribute(doc,el,newAttr,remove){
  1296. doc && doc._inc++;
  1297. var ns = newAttr.namespaceURI ;
  1298. if(ns == 'http://www.w3.org/2000/xmlns/'){
  1299. //update namespace
  1300. delete el._nsMap[newAttr.prefix?newAttr.localName:'']
  1301. }
  1302. }
  1303. function _onUpdateChild(doc,el,newChild){
  1304. if(doc && doc._inc){
  1305. doc._inc++;
  1306. //update childNodes
  1307. var cs = el.childNodes;
  1308. if(newChild){
  1309. cs[cs.length++] = newChild;
  1310. }else{
  1311. //console.log(1)
  1312. var child = el.firstChild;
  1313. var i = 0;
  1314. while(child){
  1315. cs[i++] = child;
  1316. child =child.nextSibling;
  1317. }
  1318. cs.length = i;
  1319. }
  1320. }
  1321. }
  1322. /**
  1323. * attributes;
  1324. * children;
  1325. *
  1326. * writeable properties:
  1327. * nodeValue,Attr:value,CharacterData:data
  1328. * prefix
  1329. */
  1330. function _removeChild(parentNode,child){
  1331. var previous = child.previousSibling;
  1332. var next = child.nextSibling;
  1333. if(previous){
  1334. previous.nextSibling = next;
  1335. }else{
  1336. parentNode.firstChild = next
  1337. }
  1338. if(next){
  1339. next.previousSibling = previous;
  1340. }else{
  1341. parentNode.lastChild = previous;
  1342. }
  1343. _onUpdateChild(parentNode.ownerDocument,parentNode);
  1344. return child;
  1345. }
  1346. /**
  1347. * preformance key(refChild == null)
  1348. */
  1349. function _insertBefore(parentNode,newChild,nextChild){
  1350. var cp = newChild.parentNode;
  1351. if(cp){
  1352. cp.removeChild(newChild);//remove and update
  1353. }
  1354. if(newChild.nodeType === DOCUMENT_FRAGMENT_NODE){
  1355. var newFirst = newChild.firstChild;
  1356. if (newFirst == null) {
  1357. return newChild;
  1358. }
  1359. var newLast = newChild.lastChild;
  1360. }else{
  1361. newFirst = newLast = newChild;
  1362. }
  1363. var pre = nextChild ? nextChild.previousSibling : parentNode.lastChild;
  1364. newFirst.previousSibling = pre;
  1365. newLast.nextSibling = nextChild;
  1366. if(pre){
  1367. pre.nextSibling = newFirst;
  1368. }else{
  1369. parentNode.firstChild = newFirst;
  1370. }
  1371. if(nextChild == null){
  1372. parentNode.lastChild = newLast;
  1373. }else{
  1374. nextChild.previousSibling = newLast;
  1375. }
  1376. do{
  1377. newFirst.parentNode = parentNode;
  1378. }while(newFirst !== newLast && (newFirst= newFirst.nextSibling))
  1379. _onUpdateChild(parentNode.ownerDocument||parentNode,parentNode);
  1380. //console.log(parentNode.lastChild.nextSibling == null)
  1381. if (newChild.nodeType == DOCUMENT_FRAGMENT_NODE) {
  1382. newChild.firstChild = newChild.lastChild = null;
  1383. }
  1384. return newChild;
  1385. }
  1386. function _appendSingleChild(parentNode,newChild){
  1387. var cp = newChild.parentNode;
  1388. if(cp){
  1389. var pre = parentNode.lastChild;
  1390. cp.removeChild(newChild);//remove and update
  1391. var pre = parentNode.lastChild;
  1392. }
  1393. var pre = parentNode.lastChild;
  1394. newChild.parentNode = parentNode;
  1395. newChild.previousSibling = pre;
  1396. newChild.nextSibling = null;
  1397. if(pre){
  1398. pre.nextSibling = newChild;
  1399. }else{
  1400. parentNode.firstChild = newChild;
  1401. }
  1402. parentNode.lastChild = newChild;
  1403. _onUpdateChild(parentNode.ownerDocument,parentNode,newChild);
  1404. return newChild;
  1405. //console.log("__aa",parentNode.lastChild.nextSibling == null)
  1406. }
  1407. Document.prototype = {
  1408. //implementation : null,
  1409. nodeName : '#document',
  1410. nodeType : DOCUMENT_NODE,
  1411. doctype : null,
  1412. documentElement : null,
  1413. _inc : 1,
  1414. insertBefore : function(newChild, refChild){//raises
  1415. if(newChild.nodeType == DOCUMENT_FRAGMENT_NODE){
  1416. var child = newChild.firstChild;
  1417. while(child){
  1418. var next = child.nextSibling;
  1419. this.insertBefore(child,refChild);
  1420. child = next;
  1421. }
  1422. return newChild;
  1423. }
  1424. if(this.documentElement == null && newChild.nodeType == ELEMENT_NODE){
  1425. this.documentElement = newChild;
  1426. }
  1427. return _insertBefore(this,newChild,refChild),(newChild.ownerDocument = this),newChild;
  1428. },
  1429. removeChild : function(oldChild){
  1430. if(this.documentElement == oldChild){
  1431. this.documentElement = null;
  1432. }
  1433. return _removeChild(this,oldChild);
  1434. },
  1435. // Introduced in DOM Level 2:
  1436. importNode : function(importedNode,deep){
  1437. return importNode(this,importedNode,deep);
  1438. },
  1439. // Introduced in DOM Level 2:
  1440. getElementById : function(id){
  1441. var rtv = null;
  1442. _visitNode(this.documentElement,function(node){
  1443. if(node.nodeType == ELEMENT_NODE){
  1444. if(node.getAttribute('id') == id){
  1445. rtv = node;
  1446. return true;
  1447. }
  1448. }
  1449. })
  1450. return rtv;
  1451. },
  1452. //document factory method:
  1453. createElement : function(tagName){
  1454. var node = new Element();
  1455. node.ownerDocument = this;
  1456. node.nodeName = tagName;
  1457. node.tagName = tagName;
  1458. node.childNodes = new NodeList();
  1459. var attrs = node.attributes = new NamedNodeMap();
  1460. attrs._ownerElement = node;
  1461. return node;
  1462. },
  1463. createDocumentFragment : function(){
  1464. var node = new DocumentFragment();
  1465. node.ownerDocument = this;
  1466. node.childNodes = new NodeList();
  1467. return node;
  1468. },
  1469. createTextNode : function(data){
  1470. var node = new Text();
  1471. node.ownerDocument = this;
  1472. node.appendData(data)
  1473. return node;
  1474. },
  1475. createComment : function(data){
  1476. var node = new Comment();
  1477. node.ownerDocument = this;
  1478. node.appendData(data)
  1479. return node;
  1480. },
  1481. createCDATASection : function(data){
  1482. var node = new CDATASection();
  1483. node.ownerDocument = this;
  1484. node.appendData(data)
  1485. return node;
  1486. },
  1487. createProcessingInstruction : function(target,data){
  1488. var node = new ProcessingInstruction();
  1489. node.ownerDocument = this;
  1490. node.tagName = node.target = target;
  1491. node.nodeValue= node.data = data;
  1492. return node;
  1493. },
  1494. createAttribute : function(name){
  1495. var node = new Attr();
  1496. node.ownerDocument = this;
  1497. node.name = name;
  1498. node.nodeName = name;
  1499. node.localName = name;
  1500. node.specified = true;
  1501. return node;
  1502. },
  1503. createEntityReference : function(name){
  1504. var node = new EntityReference();
  1505. node.ownerDocument = this;
  1506. node.nodeName = name;
  1507. return node;
  1508. },
  1509. // Introduced in DOM Level 2:
  1510. createElementNS : function(namespaceURI,qualifiedName){
  1511. var node = new Element();
  1512. var pl = qualifiedName.split(':');
  1513. var attrs = node.attributes = new NamedNodeMap();
  1514. node.childNodes = new NodeList();
  1515. node.ownerDocument = this;
  1516. node.nodeName = qualifiedName;
  1517. node.tagName = qualifiedName;
  1518. node.namespaceURI = namespaceURI;
  1519. if(pl.length == 2){
  1520. node.prefix = pl[0];
  1521. node.localName = pl[1];
  1522. }else{
  1523. //el.prefix = null;
  1524. node.localName = qualifiedName;
  1525. }
  1526. attrs._ownerElement = node;
  1527. return node;
  1528. },
  1529. // Introduced in DOM Level 2:
  1530. createAttributeNS : function(namespaceURI,qualifiedName){
  1531. var node = new Attr();
  1532. var pl = qualifiedName.split(':');
  1533. node.ownerDocument = this;
  1534. node.nodeName = qualifiedName;
  1535. node.name = qualifiedName;
  1536. node.namespaceURI = namespaceURI;
  1537. node.specified = true;
  1538. if(pl.length == 2){
  1539. node.prefix = pl[0];
  1540. node.localName = pl[1];
  1541. }else{
  1542. //el.prefix = null;
  1543. node.localName = qualifiedName;
  1544. }
  1545. return node;
  1546. }
  1547. };
  1548. _extends(Document,Node);
  1549. function Element() {
  1550. this._nsMap = {};
  1551. };
  1552. Element.prototype = {
  1553. nodeType : ELEMENT_NODE,
  1554. hasAttribute : function(name){
  1555. return this.getAttributeNode(name)!=null;
  1556. },
  1557. getAttribute : function(name){
  1558. var attr = this.getAttributeNode(name);
  1559. return attr && attr.value || '';
  1560. },
  1561. getAttributeNode : function(name){
  1562. return this.attributes.getNamedItem(name);
  1563. },
  1564. setAttribute : function(name, value){
  1565. var attr = this.ownerDocument.createAttribute(name);
  1566. attr.value = attr.nodeValue = "" + value;
  1567. this.setAttributeNode(attr)
  1568. },
  1569. removeAttribute : function(name){
  1570. var attr = this.getAttributeNode(name)
  1571. attr && this.removeAttributeNode(attr);
  1572. },
  1573. //four real opeartion method
  1574. appendChild:function(newChild){
  1575. if(newChild.nodeType === DOCUMENT_FRAGMENT_NODE){
  1576. return this.insertBefore(newChild,null);
  1577. }else{
  1578. return _appendSingleChild(this,newChild);
  1579. }
  1580. },
  1581. setAttributeNode : function(newAttr){
  1582. return this.attributes.setNamedItem(newAttr);
  1583. },
  1584. setAttributeNodeNS : function(newAttr){
  1585. return this.attributes.setNamedItemNS(newAttr);
  1586. },
  1587. removeAttributeNode : function(oldAttr){
  1588. //console.log(this == oldAttr.ownerElement)
  1589. return this.attributes.removeNamedItem(oldAttr.nodeName);
  1590. },
  1591. //get real attribute name,and remove it by removeAttributeNode
  1592. removeAttributeNS : function(namespaceURI, localName){
  1593. var old = this.getAttributeNodeNS(namespaceURI, localName);
  1594. old && this.removeAttributeNode(old);
  1595. },
  1596. hasAttributeNS : function(namespaceURI, localName){
  1597. return this.getAttributeNodeNS(namespaceURI, localName)!=null;
  1598. },
  1599. getAttributeNS : function(namespaceURI, localName){
  1600. var attr = this.getAttributeNodeNS(namespaceURI, localName);
  1601. return attr && attr.value || '';
  1602. },
  1603. setAttributeNS : function(namespaceURI, qualifiedName, value){
  1604. var attr = this.ownerDocument.createAttributeNS(namespaceURI, qualifiedName);
  1605. attr.value = attr.nodeValue = "" + value;
  1606. this.setAttributeNode(attr)
  1607. },
  1608. getAttributeNodeNS : function(namespaceURI, localName){
  1609. return this.attributes.getNamedItemNS(namespaceURI, localName);
  1610. },
  1611. getElementsByTagName : function(tagName){
  1612. return new LiveNodeList(this,function(base){
  1613. var ls = [];
  1614. _visitNode(base,function(node){
  1615. if(node !== base && node.nodeType == ELEMENT_NODE && (tagName === '*' || node.tagName == tagName)){
  1616. ls.push(node);
  1617. }
  1618. });
  1619. return ls;
  1620. });
  1621. },
  1622. getElementsByTagNameNS : function(namespaceURI, localName){
  1623. return new LiveNodeList(this,function(base){
  1624. var ls = [];
  1625. _visitNode(base,function(node){
  1626. if(node !== base && node.nodeType === ELEMENT_NODE && (namespaceURI === '*' || node.namespaceURI === namespaceURI) && (localName === '*' || node.localName == localName)){
  1627. ls.push(node);
  1628. }
  1629. });
  1630. return ls;
  1631. });
  1632. }
  1633. };
  1634. Document.prototype.getElementsByTagName = Element.prototype.getElementsByTagName;
  1635. Document.prototype.getElementsByTagNameNS = Element.prototype.getElementsByTagNameNS;
  1636. _extends(Element,Node);
  1637. function Attr() {
  1638. };
  1639. Attr.prototype.nodeType = ATTRIBUTE_NODE;
  1640. _extends(Attr,Node);
  1641. function CharacterData() {
  1642. };
  1643. CharacterData.prototype = {
  1644. data : '',
  1645. substringData : function(offset, count) {
  1646. return this.data.substring(offset, offset+count);
  1647. },
  1648. appendData: function(text) {
  1649. text = this.data+text;
  1650. this.nodeValue = this.data = text;
  1651. this.length = text.length;
  1652. },
  1653. insertData: function(offset,text) {
  1654. this.replaceData(offset,0,text);
  1655. },
  1656. appendChild:function(newChild){
  1657. throw new Error(ExceptionMessage[HIERARCHY_REQUEST_ERR])
  1658. },
  1659. deleteData: function(offset, count) {
  1660. this.replaceData(offset,count,"");
  1661. },
  1662. replaceData: function(offset, count, text) {
  1663. var start = this.data.substring(0,offset);
  1664. var end = this.data.substring(offset+count);
  1665. text = start + text + end;
  1666. this.nodeValue = this.data = text;
  1667. this.length = text.length;
  1668. }
  1669. }
  1670. _extends(CharacterData,Node);
  1671. function Text() {
  1672. };
  1673. Text.prototype = {
  1674. nodeName : "#text",
  1675. nodeType : TEXT_NODE,
  1676. splitText : function(offset) {
  1677. var text = this.data;
  1678. var newText = text.substring(offset);
  1679. text = text.substring(0, offset);
  1680. this.data = this.nodeValue = text;
  1681. this.length = text.length;
  1682. var newNode = this.ownerDocument.createTextNode(newText);
  1683. if(this.parentNode){
  1684. this.parentNode.insertBefore(newNode, this.nextSibling);
  1685. }
  1686. return newNode;
  1687. }
  1688. }
  1689. _extends(Text,CharacterData);
  1690. function Comment() {
  1691. };
  1692. Comment.prototype = {
  1693. nodeName : "#comment",
  1694. nodeType : COMMENT_NODE
  1695. }
  1696. _extends(Comment,CharacterData);
  1697. function CDATASection() {
  1698. };
  1699. CDATASection.prototype = {
  1700. nodeName : "#cdata-section",
  1701. nodeType : CDATA_SECTION_NODE
  1702. }
  1703. _extends(CDATASection,CharacterData);
  1704. function DocumentType() {
  1705. };
  1706. DocumentType.prototype.nodeType = DOCUMENT_TYPE_NODE;
  1707. _extends(DocumentType,Node);
  1708. function Notation() {
  1709. };
  1710. Notation.prototype.nodeType = NOTATION_NODE;
  1711. _extends(Notation,Node);
  1712. function Entity() {
  1713. };
  1714. Entity.prototype.nodeType = ENTITY_NODE;
  1715. _extends(Entity,Node);
  1716. function EntityReference() {
  1717. };
  1718. EntityReference.prototype.nodeType = ENTITY_REFERENCE_NODE;
  1719. _extends(EntityReference,Node);
  1720. function DocumentFragment() {
  1721. };
  1722. DocumentFragment.prototype.nodeName = "#document-fragment";
  1723. DocumentFragment.prototype.nodeType = DOCUMENT_FRAGMENT_NODE;
  1724. _extends(DocumentFragment,Node);
  1725. function ProcessingInstruction() {
  1726. }
  1727. ProcessingInstruction.prototype.nodeType = PROCESSING_INSTRUCTION_NODE;
  1728. _extends(ProcessingInstruction,Node);
  1729. function XMLSerializer(){}
  1730. XMLSerializer.prototype.serializeToString = function(node,isHtml,nodeFilter){
  1731. return nodeSerializeToString.call(node,isHtml,nodeFilter);
  1732. }
  1733. Node.prototype.toString = nodeSerializeToString;
  1734. function nodeSerializeToString(isHtml,nodeFilter){
  1735. var buf = [];
  1736. var refNode = this.nodeType == 9?this.documentElement:this;
  1737. var prefix = refNode.prefix;
  1738. var uri = refNode.namespaceURI;
  1739. if(uri && prefix == null){
  1740. //console.log(prefix)
  1741. var prefix = refNode.lookupPrefix(uri);
  1742. if(prefix == null){
  1743. //isHTML = true;
  1744. var visibleNamespaces=[
  1745. {namespace:uri,prefix:null}
  1746. //{namespace:uri,prefix:''}
  1747. ]
  1748. }
  1749. }
  1750. serializeToString(this,buf,isHtml,nodeFilter,visibleNamespaces);
  1751. //console.log('###',this.nodeType,uri,prefix,buf.join(''))
  1752. return buf.join('');
  1753. }
  1754. function needNamespaceDefine(node,isHTML, visibleNamespaces) {
  1755. var prefix = node.prefix||'';
  1756. var uri = node.namespaceURI;
  1757. if (!prefix && !uri){
  1758. return false;
  1759. }
  1760. if (prefix === "xml" && uri === "http://www.w3.org/XML/1998/namespace"
  1761. || uri == 'http://www.w3.org/2000/xmlns/'){
  1762. return false;
  1763. }
  1764. var i = visibleNamespaces.length
  1765. //console.log('@@@@',node.tagName,prefix,uri,visibleNamespaces)
  1766. while (i--) {
  1767. var ns = visibleNamespaces[i];
  1768. // get namespace prefix
  1769. //console.log(node.nodeType,node.tagName,ns.prefix,prefix)
  1770. if (ns.prefix == prefix){
  1771. return ns.namespace != uri;
  1772. }
  1773. }
  1774. //console.log(isHTML,uri,prefix=='')
  1775. //if(isHTML && prefix ==null && uri == 'http://www.w3.org/1999/xhtml'){
  1776. // return false;
  1777. //}
  1778. //node.flag = '11111'
  1779. //console.error(3,true,node.flag,node.prefix,node.namespaceURI)
  1780. return true;
  1781. }
  1782. function serializeToString(node,buf,isHTML,nodeFilter,visibleNamespaces){
  1783. if(nodeFilter){
  1784. node = nodeFilter(node);
  1785. if(node){
  1786. if(typeof node == 'string'){
  1787. buf.push(node);
  1788. return;
  1789. }
  1790. }else{
  1791. return;
  1792. }
  1793. //buf.sort.apply(attrs, attributeSorter);
  1794. }
  1795. switch(node.nodeType){
  1796. case ELEMENT_NODE:
  1797. if (!visibleNamespaces) visibleNamespaces = [];
  1798. var startVisibleNamespaces = visibleNamespaces.length;
  1799. var attrs = node.attributes;
  1800. var len = attrs.length;
  1801. var child = node.firstChild;
  1802. var nodeName = node.tagName;
  1803. isHTML = (htmlns === node.namespaceURI) ||isHTML
  1804. buf.push('<',nodeName);
  1805. for(var i=0;i<len;i++){
  1806. // add namespaces for attributes
  1807. var attr = attrs.item(i);
  1808. if (attr.prefix == 'xmlns') {
  1809. visibleNamespaces.push({ prefix: attr.localName, namespace: attr.value });
  1810. }else if(attr.nodeName == 'xmlns'){
  1811. visibleNamespaces.push({ prefix: '', namespace: attr.value });
  1812. }
  1813. }
  1814. for(var i=0;i<len;i++){
  1815. var attr = attrs.item(i);
  1816. if (needNamespaceDefine(attr,isHTML, visibleNamespaces)) {
  1817. var prefix = attr.prefix||'';
  1818. var uri = attr.namespaceURI;
  1819. var ns = prefix ? ' xmlns:' + prefix : " xmlns";
  1820. buf.push(ns, '="' , uri , '"');
  1821. visibleNamespaces.push({ prefix: prefix, namespace:uri });
  1822. }
  1823. serializeToString(attr,buf,isHTML,nodeFilter,visibleNamespaces);
  1824. }
  1825. // add namespace for current node
  1826. if (needNamespaceDefine(node,isHTML, visibleNamespaces)) {
  1827. var prefix = node.prefix||'';
  1828. var uri = node.namespaceURI;
  1829. var ns = prefix ? ' xmlns:' + prefix : " xmlns";
  1830. buf.push(ns, '="' , uri , '"');
  1831. visibleNamespaces.push({ prefix: prefix, namespace:uri });
  1832. }
  1833. if(child || isHTML && !/^(?:meta|link|img|br|hr|input)$/i.test(nodeName)){
  1834. buf.push('>');
  1835. //if is cdata child node
  1836. if(isHTML && /^script$/i.test(nodeName)){
  1837. while(child){
  1838. if(child.data){
  1839. buf.push(child.data);
  1840. }else{
  1841. serializeToString(child,buf,isHTML,nodeFilter,visibleNamespaces);
  1842. }
  1843. child = child.nextSibling;
  1844. }
  1845. }else
  1846. {
  1847. while(child){
  1848. serializeToString(child,buf,isHTML,nodeFilter,visibleNamespaces);
  1849. child = child.nextSibling;
  1850. }
  1851. }
  1852. buf.push('</',nodeName,'>');
  1853. }else{
  1854. buf.push('/>');
  1855. }
  1856. // remove added visible namespaces
  1857. //visibleNamespaces.length = startVisibleNamespaces;
  1858. return;
  1859. case DOCUMENT_NODE:
  1860. case DOCUMENT_FRAGMENT_NODE:
  1861. var child = node.firstChild;
  1862. while(child){
  1863. serializeToString(child,buf,isHTML,nodeFilter,visibleNamespaces);
  1864. child = child.nextSibling;
  1865. }
  1866. return;
  1867. case ATTRIBUTE_NODE:
  1868. return buf.push(' ',node.name,'="',node.value.replace(/[<&"]/g,_xmlEncoder),'"');
  1869. case TEXT_NODE:
  1870. return buf.push(node.data.replace(/[<&]/g,_xmlEncoder));
  1871. case CDATA_SECTION_NODE:
  1872. return buf.push( '<![CDATA[',node.data,']]>');
  1873. case COMMENT_NODE:
  1874. return buf.push( "<!--",node.data,"-->");
  1875. case DOCUMENT_TYPE_NODE:
  1876. var pubid = node.publicId;
  1877. var sysid = node.systemId;
  1878. buf.push('<!DOCTYPE ',node.name);
  1879. if(pubid){
  1880. buf.push(' PUBLIC "',pubid);
  1881. if (sysid && sysid!='.') {
  1882. buf.push( '" "',sysid);
  1883. }
  1884. buf.push('">');
  1885. }else if(sysid && sysid!='.'){
  1886. buf.push(' SYSTEM "',sysid,'">');
  1887. }else{
  1888. var sub = node.internalSubset;
  1889. if(sub){
  1890. buf.push(" [",sub,"]");
  1891. }
  1892. buf.push(">");
  1893. }
  1894. return;
  1895. case PROCESSING_INSTRUCTION_NODE:
  1896. return buf.push( "<?",node.target," ",node.data,"?>");
  1897. case ENTITY_REFERENCE_NODE:
  1898. return buf.push( '&',node.nodeName,';');
  1899. //case ENTITY_NODE:
  1900. //case NOTATION_NODE:
  1901. default:
  1902. buf.push('??',node.nodeName);
  1903. }
  1904. }
  1905. function importNode(doc,node,deep){
  1906. var node2;
  1907. switch (node.nodeType) {
  1908. case ELEMENT_NODE:
  1909. node2 = node.cloneNode(false);
  1910. node2.ownerDocument = doc;
  1911. //var attrs = node2.attributes;
  1912. //var len = attrs.length;
  1913. //for(var i=0;i<len;i++){
  1914. //node2.setAttributeNodeNS(importNode(doc,attrs.item(i),deep));
  1915. //}
  1916. case DOCUMENT_FRAGMENT_NODE:
  1917. break;
  1918. case ATTRIBUTE_NODE:
  1919. deep = true;
  1920. break;
  1921. //case ENTITY_REFERENCE_NODE:
  1922. //case PROCESSING_INSTRUCTION_NODE:
  1923. ////case TEXT_NODE:
  1924. //case CDATA_SECTION_NODE:
  1925. //case COMMENT_NODE:
  1926. // deep = false;
  1927. // break;
  1928. //case DOCUMENT_NODE:
  1929. //case DOCUMENT_TYPE_NODE:
  1930. //cannot be imported.
  1931. //case ENTITY_NODE:
  1932. //case NOTATION_NODE:
  1933. //can not hit in level3
  1934. //default:throw e;
  1935. }
  1936. if(!node2){
  1937. node2 = node.cloneNode(false);//false
  1938. }
  1939. node2.ownerDocument = doc;
  1940. node2.parentNode = null;
  1941. if(deep){
  1942. var child = node.firstChild;
  1943. while(child){
  1944. node2.appendChild(importNode(doc,child,deep));
  1945. child = child.nextSibling;
  1946. }
  1947. }
  1948. return node2;
  1949. }
  1950. //
  1951. //var _relationMap = {firstChild:1,lastChild:1,previousSibling:1,nextSibling:1,
  1952. // attributes:1,childNodes:1,parentNode:1,documentElement:1,doctype,};
  1953. function cloneNode(doc,node,deep){
  1954. var node2 = new node.constructor();
  1955. for(var n in node){
  1956. var v = node[n];
  1957. if(typeof v != 'object' ){
  1958. if(v != node2[n]){
  1959. node2[n] = v;
  1960. }
  1961. }
  1962. }
  1963. if(node.childNodes){
  1964. node2.childNodes = new NodeList();
  1965. }
  1966. node2.ownerDocument = doc;
  1967. switch (node2.nodeType) {
  1968. case ELEMENT_NODE:
  1969. var attrs = node.attributes;
  1970. var attrs2 = node2.attributes = new NamedNodeMap();
  1971. var len = attrs.length
  1972. attrs2._ownerElement = node2;
  1973. for(var i=0;i<len;i++){
  1974. node2.setAttributeNode(cloneNode(doc,attrs.item(i),true));
  1975. }
  1976. break;;
  1977. case ATTRIBUTE_NODE:
  1978. deep = true;
  1979. }
  1980. if(deep){
  1981. var child = node.firstChild;
  1982. while(child){
  1983. node2.appendChild(cloneNode(doc,child,deep));
  1984. child = child.nextSibling;
  1985. }
  1986. }
  1987. return node2;
  1988. }
  1989. function __set__(object,key,value){
  1990. object[key] = value
  1991. }
  1992. //do dynamic
  1993. try{
  1994. if(Object.defineProperty){
  1995. Object.defineProperty(LiveNodeList.prototype,'length',{
  1996. get:function(){
  1997. _updateLiveList(this);
  1998. return this.$$length;
  1999. }
  2000. });
  2001. Object.defineProperty(Node.prototype,'textContent',{
  2002. get:function(){
  2003. return getTextContent(this);
  2004. },
  2005. set:function(data){
  2006. switch(this.nodeType){
  2007. case ELEMENT_NODE:
  2008. case DOCUMENT_FRAGMENT_NODE:
  2009. while(this.firstChild){
  2010. this.removeChild(this.firstChild);
  2011. }
  2012. if(data || String(data)){
  2013. this.appendChild(this.ownerDocument.createTextNode(data));
  2014. }
  2015. break;
  2016. default:
  2017. //TODO:
  2018. this.data = data;
  2019. this.value = data;
  2020. this.nodeValue = data;
  2021. }
  2022. }
  2023. })
  2024. function getTextContent(node){
  2025. switch(node.nodeType){
  2026. case ELEMENT_NODE:
  2027. case DOCUMENT_FRAGMENT_NODE:
  2028. var buf = [];
  2029. node = node.firstChild;
  2030. while(node){
  2031. if(node.nodeType!==7 && node.nodeType !==8){
  2032. buf.push(getTextContent(node));
  2033. }
  2034. node = node.nextSibling;
  2035. }
  2036. return buf.join('');
  2037. default:
  2038. return node.nodeValue;
  2039. }
  2040. }
  2041. __set__ = function(object,key,value){
  2042. //console.log(value)
  2043. object['$$'+key] = value
  2044. }
  2045. }
  2046. }catch(e){//ie8
  2047. }
  2048. //if(typeof require == 'function'){
  2049. exports.DOMImplementation = DOMImplementation;
  2050. exports.XMLSerializer = XMLSerializer;
  2051. //}
  2052. /***/ }),
  2053. /* 3 */
  2054. /***/ (function(module, exports) {
  2055. var initEvent = function (cos) {
  2056. var listeners = {};
  2057. var getList = function (action) {
  2058. !listeners[action] && (listeners[action] = []);
  2059. return listeners[action];
  2060. };
  2061. cos.on = function (action, callback) {
  2062. if (action === 'task-list-update') {
  2063. console.warn('warning: Event "' + action + '" has been deprecated. Please use "list-update" instead.');
  2064. }
  2065. getList(action).push(callback);
  2066. };
  2067. cos.off = function (action, callback) {
  2068. var list = getList(action);
  2069. for (var i = list.length - 1; i >= 0; i--) {
  2070. callback === list[i] && list.splice(i, 1);
  2071. }
  2072. };
  2073. cos.emit = function (action, data) {
  2074. var list = getList(action).map(function (cb) {
  2075. return cb;
  2076. });
  2077. for (var i = 0; i < list.length; i++) {
  2078. list[i](data);
  2079. }
  2080. };
  2081. };
  2082. var EventProxy = function () {
  2083. initEvent(this);
  2084. };
  2085. module.exports.init = initEvent;
  2086. module.exports.EventProxy = EventProxy;
  2087. /***/ }),
  2088. /* 4 */
  2089. /***/ (function(module, exports, __webpack_require__) {
  2090. var util = __webpack_require__(0);
  2091. // 按照文件特征值,缓存 UploadId
  2092. var cacheKey = 'cos_sdk_upload_cache';
  2093. var expires = 30 * 24 * 3600;
  2094. var cache;
  2095. var timer;
  2096. var getCache = function () {
  2097. try {
  2098. var val = JSON.parse(localStorage.getItem(cacheKey));
  2099. } catch (e) {}
  2100. if (!val) val = [];
  2101. cache = val;
  2102. };
  2103. var setCache = function () {
  2104. try {
  2105. localStorage.setItem(cacheKey, JSON.stringify(cache));
  2106. } catch (e) {}
  2107. };
  2108. var init = function () {
  2109. if (cache) return;
  2110. getCache.call(this);
  2111. // 清理太老旧的数据
  2112. var changed = false;
  2113. var now = Math.round(Date.now() / 1000);
  2114. for (var i = cache.length - 1; i >= 0; i--) {
  2115. var mtime = cache[i][2];
  2116. if (!mtime || mtime + expires < now) {
  2117. cache.splice(i, 1);
  2118. changed = true;
  2119. }
  2120. }
  2121. changed && setCache();
  2122. };
  2123. // 把缓存存到本地
  2124. var save = function () {
  2125. if (timer) return;
  2126. timer = setTimeout(function () {
  2127. setCache();
  2128. timer = null;
  2129. }, 400);
  2130. };
  2131. var mod = {
  2132. using: {},
  2133. // 标记 UploadId 正在使用
  2134. setUsing: function (uuid) {
  2135. mod.using[uuid] = true;
  2136. },
  2137. // 标记 UploadId 已经没在使用
  2138. removeUsing: function (uuid) {
  2139. delete mod.using[uuid];
  2140. },
  2141. // 用上传参数生成哈希值
  2142. getFileId: function (file, ChunkSize, Bucket, Key) {
  2143. if (file.name && file.size && file.lastModifiedDate && ChunkSize) {
  2144. return util.md5([file.name, file.size, file.lastModifiedDate, ChunkSize, Bucket, Key].join('::'));
  2145. } else {
  2146. return null;
  2147. }
  2148. },
  2149. // 获取文件对应的 UploadId 列表
  2150. getUploadIdList: function (uuid) {
  2151. if (!uuid) return null;
  2152. init.call(this);
  2153. var list = [];
  2154. for (var i = 0; i < cache.length; i++) {
  2155. if (cache[i][0] === uuid) list.push(cache[i][1]);
  2156. }
  2157. return list.length ? list : null;
  2158. },
  2159. // 缓存 UploadId
  2160. saveUploadId: function (uuid, UploadId, limit) {
  2161. init.call(this);
  2162. if (!uuid) return;
  2163. // 清理没用的 UploadId,js 文件没有 FilePath ,只清理相同记录
  2164. for (var i = cache.length - 1; i >= 0; i--) {
  2165. var item = cache[i];
  2166. if (item[0] === uuid && item[1] === UploadId) {
  2167. cache.splice(i, 1);
  2168. }
  2169. }
  2170. cache.unshift([uuid, UploadId, Math.round(Date.now() / 1000)]);
  2171. if (cache.length > limit) cache.splice(limit);
  2172. save();
  2173. },
  2174. // UploadId 已用完,移除掉
  2175. removeUploadId: function (UploadId) {
  2176. init.call(this);
  2177. delete mod.using[UploadId];
  2178. for (var i = cache.length - 1; i >= 0; i--) {
  2179. if (cache[i][1] === UploadId) cache.splice(i, 1);
  2180. }
  2181. save();
  2182. }
  2183. };
  2184. module.exports = mod;
  2185. /***/ }),
  2186. /* 5 */
  2187. /***/ (function(module, exports, __webpack_require__) {
  2188. var COS = __webpack_require__(6);
  2189. module.exports = COS;
  2190. /***/ }),
  2191. /* 6 */
  2192. /***/ (function(module, exports, __webpack_require__) {
  2193. "use strict";
  2194. var util = __webpack_require__(0);
  2195. var event = __webpack_require__(3);
  2196. var task = __webpack_require__(15);
  2197. var base = __webpack_require__(16);
  2198. var advance = __webpack_require__(18);
  2199. var defaultOptions = {
  2200. AppId: '', // AppId 已废弃,请拼接到 Bucket 后传入,例如:test-1250000000
  2201. SecretId: '',
  2202. SecretKey: '',
  2203. SecurityToken: '', // 使用临时密钥需要注意自行刷新 Token
  2204. ChunkRetryTimes: 2,
  2205. FileParallelLimit: 3,
  2206. ChunkParallelLimit: 3,
  2207. ChunkSize: 1024 * 1024,
  2208. SliceSize: 1024 * 1024,
  2209. CopyChunkParallelLimit: 20,
  2210. CopyChunkSize: 1024 * 1024 * 10,
  2211. CopySliceSize: 1024 * 1024 * 10,
  2212. MaxPartNumber: 10000,
  2213. ProgressInterval: 1000,
  2214. Domain: '',
  2215. ServiceDomain: '',
  2216. Protocol: '',
  2217. CompatibilityMode: false,
  2218. ForcePathStyle: false,
  2219. UseRawKey: false,
  2220. Timeout: 0, // 单位毫秒,0 代表不设置超时时间
  2221. CorrectClockSkew: true,
  2222. SystemClockOffset: 0, // 单位毫秒,ms
  2223. UploadCheckContentMd5: false,
  2224. UploadQueueSize: 10000,
  2225. UploadAddMetaMd5: false,
  2226. UploadIdCacheLimit: 50,
  2227. UseAccelerate: false
  2228. };
  2229. // 对外暴露的类
  2230. var COS = function (options) {
  2231. this.options = util.extend(util.clone(defaultOptions), options || {});
  2232. this.options.FileParallelLimit = Math.max(1, this.options.FileParallelLimit);
  2233. this.options.ChunkParallelLimit = Math.max(1, this.options.ChunkParallelLimit);
  2234. this.options.ChunkRetryTimes = Math.max(0, this.options.ChunkRetryTimes);
  2235. this.options.ChunkSize = Math.max(1024 * 1024, this.options.ChunkSize);
  2236. this.options.CopyChunkParallelLimit = Math.max(1, this.options.CopyChunkParallelLimit);
  2237. this.options.CopyChunkSize = Math.max(1024 * 1024, this.options.CopyChunkSize);
  2238. this.options.CopySliceSize = Math.max(0, this.options.CopySliceSize);
  2239. this.options.MaxPartNumber = Math.max(1024, Math.min(10000, this.options.MaxPartNumber));
  2240. this.options.Timeout = Math.max(0, this.options.Timeout);
  2241. if (this.options.AppId) {
  2242. console.warn('warning: AppId has been deprecated, Please put it at the end of parameter Bucket(E.g: "test-1250000000").');
  2243. }
  2244. if (util.isNode()) {
  2245. console.warn('warning: cos-js-sdk-v5 不支持 nodejs 环境使用,请改用 cos-nodejs-sdk-v5,参考文档: https://cloud.tencent.com/document/product/436/8629');
  2246. console.warn('warning: cos-js-sdk-v5 does not support nodejs environment. Please use cos-nodejs-sdk-v5 instead. See: https://cloud.tencent.com/document/product/436/8629');
  2247. }
  2248. event.init(this);
  2249. task.init(this);
  2250. };
  2251. base.init(COS, task);
  2252. advance.init(COS, task);
  2253. COS.getAuthorization = util.getAuth;
  2254. COS.version = '1.2.14';
  2255. module.exports = COS;
  2256. /***/ }),
  2257. /* 7 */
  2258. /***/ (function(module, exports, __webpack_require__) {
  2259. /* WEBPACK VAR INJECTION */(function(process, global) {var __WEBPACK_AMD_DEFINE_RESULT__;/* https://github.com/emn178/js-md5 */
  2260. (function () {
  2261. 'use strict';
  2262. var ERROR = 'input is invalid type';
  2263. var WINDOW = typeof window === 'object';
  2264. var root = WINDOW ? window : {};
  2265. if (root.JS_MD5_NO_WINDOW) {
  2266. WINDOW = false;
  2267. }
  2268. var WEB_WORKER = !WINDOW && typeof self === 'object';
  2269. var NODE_JS = !root.JS_MD5_NO_NODE_JS && typeof process === 'object' && process.versions && process.versions.node;
  2270. if (NODE_JS) {
  2271. root = global;
  2272. } else if (WEB_WORKER) {
  2273. root = self;
  2274. }
  2275. var COMMON_JS = !root.JS_MD5_NO_COMMON_JS && typeof module === 'object' && module.exports;
  2276. var AMD = "function" === 'function' && __webpack_require__(9);
  2277. var ARRAY_BUFFER = !root.JS_MD5_NO_ARRAY_BUFFER && typeof ArrayBuffer !== 'undefined';
  2278. var HEX_CHARS = '0123456789abcdef'.split('');
  2279. var EXTRA = [128, 32768, 8388608, -2147483648];
  2280. var SHIFT = [0, 8, 16, 24];
  2281. var OUTPUT_TYPES = ['hex', 'array', 'digest', 'buffer', 'arrayBuffer', 'base64'];
  2282. var BASE64_ENCODE_CHAR = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/'.split('');
  2283. var blocks = [],
  2284. buffer8;
  2285. if (ARRAY_BUFFER) {
  2286. var buffer = new ArrayBuffer(68);
  2287. buffer8 = new Uint8Array(buffer);
  2288. blocks = new Uint32Array(buffer);
  2289. }
  2290. if (root.JS_MD5_NO_NODE_JS || !Array.isArray) {
  2291. Array.isArray = function (obj) {
  2292. return Object.prototype.toString.call(obj) === '[object Array]';
  2293. };
  2294. }
  2295. if (ARRAY_BUFFER && (root.JS_MD5_NO_ARRAY_BUFFER_IS_VIEW || !ArrayBuffer.isView)) {
  2296. ArrayBuffer.isView = function (obj) {
  2297. return typeof obj === 'object' && obj.buffer && obj.buffer.constructor === ArrayBuffer;
  2298. };
  2299. }
  2300. /**
  2301. * @method hex
  2302. * @memberof md5
  2303. * @description Output hash as hex string
  2304. * @param {String|Array|Uint8Array|ArrayBuffer} message message to hash
  2305. * @returns {String} Hex string
  2306. * @example
  2307. * md5.hex('The quick brown fox jumps over the lazy dog');
  2308. * // equal to
  2309. * md5('The quick brown fox jumps over the lazy dog');
  2310. */
  2311. /**
  2312. * @method digest
  2313. * @memberof md5
  2314. * @description Output hash as bytes array
  2315. * @param {String|Array|Uint8Array|ArrayBuffer} message message to hash
  2316. * @returns {Array} Bytes array
  2317. * @example
  2318. * md5.digest('The quick brown fox jumps over the lazy dog');
  2319. */
  2320. /**
  2321. * @method array
  2322. * @memberof md5
  2323. * @description Output hash as bytes array
  2324. * @param {String|Array|Uint8Array|ArrayBuffer} message message to hash
  2325. * @returns {Array} Bytes array
  2326. * @example
  2327. * md5.array('The quick brown fox jumps over the lazy dog');
  2328. */
  2329. /**
  2330. * @method arrayBuffer
  2331. * @memberof md5
  2332. * @description Output hash as ArrayBuffer
  2333. * @param {String|Array|Uint8Array|ArrayBuffer} message message to hash
  2334. * @returns {ArrayBuffer} ArrayBuffer
  2335. * @example
  2336. * md5.arrayBuffer('The quick brown fox jumps over the lazy dog');
  2337. */
  2338. /**
  2339. * @method buffer
  2340. * @deprecated This maybe confuse with Buffer in node.js. Please use arrayBuffer instead.
  2341. * @memberof md5
  2342. * @description Output hash as ArrayBuffer
  2343. * @param {String|Array|Uint8Array|ArrayBuffer} message message to hash
  2344. * @returns {ArrayBuffer} ArrayBuffer
  2345. * @example
  2346. * md5.buffer('The quick brown fox jumps over the lazy dog');
  2347. */
  2348. /**
  2349. * @method base64
  2350. * @memberof md5
  2351. * @description Output hash as base64 string
  2352. * @param {String|Array|Uint8Array|ArrayBuffer} message message to hash
  2353. * @returns {String} base64 string
  2354. * @example
  2355. * md5.base64('The quick brown fox jumps over the lazy dog');
  2356. */
  2357. var createOutputMethod = function (outputType) {
  2358. return function (message, isBinStr) {
  2359. return new Md5(true).update(message, isBinStr)[outputType]();
  2360. };
  2361. };
  2362. /**
  2363. * @method create
  2364. * @memberof md5
  2365. * @description Create Md5 object
  2366. * @returns {Md5} Md5 object.
  2367. * @example
  2368. * var hash = md5.create();
  2369. */
  2370. /**
  2371. * @method update
  2372. * @memberof md5
  2373. * @description Create and update Md5 object
  2374. * @param {String|Array|Uint8Array|ArrayBuffer} message message to hash
  2375. * @returns {Md5} Md5 object.
  2376. * @example
  2377. * var hash = md5.update('The quick brown fox jumps over the lazy dog');
  2378. * // equal to
  2379. * var hash = md5.create();
  2380. * hash.update('The quick brown fox jumps over the lazy dog');
  2381. */
  2382. var createMethod = function () {
  2383. var method = createOutputMethod('hex');
  2384. if (NODE_JS) {
  2385. method = nodeWrap(method);
  2386. }
  2387. method.getCtx = method.create = function () {
  2388. return new Md5();
  2389. };
  2390. method.update = function (message) {
  2391. return method.create().update(message);
  2392. };
  2393. for (var i = 0; i < OUTPUT_TYPES.length; ++i) {
  2394. var type = OUTPUT_TYPES[i];
  2395. method[type] = createOutputMethod(type);
  2396. }
  2397. return method;
  2398. };
  2399. var nodeWrap = function (method) {
  2400. var crypto = eval("require('crypto')");
  2401. var Buffer = eval("require('buffer').Buffer");
  2402. var nodeMethod = function (message) {
  2403. if (typeof message === 'string') {
  2404. return crypto.createHash('md5').update(message, 'utf8').digest('hex');
  2405. } else {
  2406. if (message === null || message === undefined) {
  2407. throw ERROR;
  2408. } else if (message.constructor === ArrayBuffer) {
  2409. message = new Uint8Array(message);
  2410. }
  2411. }
  2412. if (Array.isArray(message) || ArrayBuffer.isView(message) || message.constructor === Buffer) {
  2413. return crypto.createHash('md5').update(new Buffer(message)).digest('hex');
  2414. } else {
  2415. return method(message);
  2416. }
  2417. };
  2418. return nodeMethod;
  2419. };
  2420. /**
  2421. * Md5 class
  2422. * @class Md5
  2423. * @description This is internal class.
  2424. * @see {@link md5.create}
  2425. */
  2426. function Md5(sharedMemory) {
  2427. if (sharedMemory) {
  2428. blocks[0] = blocks[16] = blocks[1] = blocks[2] = blocks[3] = blocks[4] = blocks[5] = blocks[6] = blocks[7] = blocks[8] = blocks[9] = blocks[10] = blocks[11] = blocks[12] = blocks[13] = blocks[14] = blocks[15] = 0;
  2429. this.blocks = blocks;
  2430. this.buffer8 = buffer8;
  2431. } else {
  2432. if (ARRAY_BUFFER) {
  2433. var buffer = new ArrayBuffer(68);
  2434. this.buffer8 = new Uint8Array(buffer);
  2435. this.blocks = new Uint32Array(buffer);
  2436. } else {
  2437. this.blocks = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0];
  2438. }
  2439. }
  2440. this.h0 = this.h1 = this.h2 = this.h3 = this.start = this.bytes = this.hBytes = 0;
  2441. this.finalized = this.hashed = false;
  2442. this.first = true;
  2443. }
  2444. /**
  2445. * @method update
  2446. * @memberof Md5
  2447. * @instance
  2448. * @description Update hash
  2449. * @param {String|Array|Uint8Array|ArrayBuffer} message message to hash
  2450. * @returns {Md5} Md5 object.
  2451. * @see {@link md5.update}
  2452. */
  2453. Md5.prototype.update = function (message, isBinStr) {
  2454. if (this.finalized) {
  2455. return;
  2456. }
  2457. var code,
  2458. index = 0,
  2459. i,
  2460. length = message.length,
  2461. blocks = this.blocks;
  2462. var buffer8 = this.buffer8;
  2463. while (index < length) {
  2464. if (this.hashed) {
  2465. this.hashed = false;
  2466. blocks[0] = blocks[16];
  2467. blocks[16] = blocks[1] = blocks[2] = blocks[3] = blocks[4] = blocks[5] = blocks[6] = blocks[7] = blocks[8] = blocks[9] = blocks[10] = blocks[11] = blocks[12] = blocks[13] = blocks[14] = blocks[15] = 0;
  2468. }
  2469. if (ARRAY_BUFFER) {
  2470. for (i = this.start; index < length && i < 64; ++index) {
  2471. code = message.charCodeAt(index);
  2472. if (isBinStr || code < 0x80) {
  2473. buffer8[i++] = code;
  2474. } else if (code < 0x800) {
  2475. buffer8[i++] = 0xc0 | code >> 6;
  2476. buffer8[i++] = 0x80 | code & 0x3f;
  2477. } else if (code < 0xd800 || code >= 0xe000) {
  2478. buffer8[i++] = 0xe0 | code >> 12;
  2479. buffer8[i++] = 0x80 | code >> 6 & 0x3f;
  2480. buffer8[i++] = 0x80 | code & 0x3f;
  2481. } else {
  2482. code = 0x10000 + ((code & 0x3ff) << 10 | message.charCodeAt(++index) & 0x3ff);
  2483. buffer8[i++] = 0xf0 | code >> 18;
  2484. buffer8[i++] = 0x80 | code >> 12 & 0x3f;
  2485. buffer8[i++] = 0x80 | code >> 6 & 0x3f;
  2486. buffer8[i++] = 0x80 | code & 0x3f;
  2487. }
  2488. }
  2489. } else {
  2490. for (i = this.start; index < length && i < 64; ++index) {
  2491. code = message.charCodeAt(index);
  2492. if (isBinStr || code < 0x80) {
  2493. blocks[i >> 2] |= code << SHIFT[i++ & 3];
  2494. } else if (code < 0x800) {
  2495. blocks[i >> 2] |= (0xc0 | code >> 6) << SHIFT[i++ & 3];
  2496. blocks[i >> 2] |= (0x80 | code & 0x3f) << SHIFT[i++ & 3];
  2497. } else if (code < 0xd800 || code >= 0xe000) {
  2498. blocks[i >> 2] |= (0xe0 | code >> 12) << SHIFT[i++ & 3];
  2499. blocks[i >> 2] |= (0x80 | code >> 6 & 0x3f) << SHIFT[i++ & 3];
  2500. blocks[i >> 2] |= (0x80 | code & 0x3f) << SHIFT[i++ & 3];
  2501. } else {
  2502. code = 0x10000 + ((code & 0x3ff) << 10 | message.charCodeAt(++index) & 0x3ff);
  2503. blocks[i >> 2] |= (0xf0 | code >> 18) << SHIFT[i++ & 3];
  2504. blocks[i >> 2] |= (0x80 | code >> 12 & 0x3f) << SHIFT[i++ & 3];
  2505. blocks[i >> 2] |= (0x80 | code >> 6 & 0x3f) << SHIFT[i++ & 3];
  2506. blocks[i >> 2] |= (0x80 | code & 0x3f) << SHIFT[i++ & 3];
  2507. }
  2508. }
  2509. }
  2510. this.lastByteIndex = i;
  2511. this.bytes += i - this.start;
  2512. if (i >= 64) {
  2513. this.start = i - 64;
  2514. this.hash();
  2515. this.hashed = true;
  2516. } else {
  2517. this.start = i;
  2518. }
  2519. }
  2520. if (this.bytes > 4294967295) {
  2521. this.hBytes += this.bytes / 4294967296 << 0;
  2522. this.bytes = this.bytes % 4294967296;
  2523. }
  2524. return this;
  2525. };
  2526. Md5.prototype.finalize = function () {
  2527. if (this.finalized) {
  2528. return;
  2529. }
  2530. this.finalized = true;
  2531. var blocks = this.blocks,
  2532. i = this.lastByteIndex;
  2533. blocks[i >> 2] |= EXTRA[i & 3];
  2534. if (i >= 56) {
  2535. if (!this.hashed) {
  2536. this.hash();
  2537. }
  2538. blocks[0] = blocks[16];
  2539. blocks[16] = blocks[1] = blocks[2] = blocks[3] = blocks[4] = blocks[5] = blocks[6] = blocks[7] = blocks[8] = blocks[9] = blocks[10] = blocks[11] = blocks[12] = blocks[13] = blocks[14] = blocks[15] = 0;
  2540. }
  2541. blocks[14] = this.bytes << 3;
  2542. blocks[15] = this.hBytes << 3 | this.bytes >>> 29;
  2543. this.hash();
  2544. };
  2545. Md5.prototype.hash = function () {
  2546. var a,
  2547. b,
  2548. c,
  2549. d,
  2550. bc,
  2551. da,
  2552. blocks = this.blocks;
  2553. if (this.first) {
  2554. a = blocks[0] - 680876937;
  2555. a = (a << 7 | a >>> 25) - 271733879 << 0;
  2556. d = (-1732584194 ^ a & 2004318071) + blocks[1] - 117830708;
  2557. d = (d << 12 | d >>> 20) + a << 0;
  2558. c = (-271733879 ^ d & (a ^ -271733879)) + blocks[2] - 1126478375;
  2559. c = (c << 17 | c >>> 15) + d << 0;
  2560. b = (a ^ c & (d ^ a)) + blocks[3] - 1316259209;
  2561. b = (b << 22 | b >>> 10) + c << 0;
  2562. } else {
  2563. a = this.h0;
  2564. b = this.h1;
  2565. c = this.h2;
  2566. d = this.h3;
  2567. a += (d ^ b & (c ^ d)) + blocks[0] - 680876936;
  2568. a = (a << 7 | a >>> 25) + b << 0;
  2569. d += (c ^ a & (b ^ c)) + blocks[1] - 389564586;
  2570. d = (d << 12 | d >>> 20) + a << 0;
  2571. c += (b ^ d & (a ^ b)) + blocks[2] + 606105819;
  2572. c = (c << 17 | c >>> 15) + d << 0;
  2573. b += (a ^ c & (d ^ a)) + blocks[3] - 1044525330;
  2574. b = (b << 22 | b >>> 10) + c << 0;
  2575. }
  2576. a += (d ^ b & (c ^ d)) + blocks[4] - 176418897;
  2577. a = (a << 7 | a >>> 25) + b << 0;
  2578. d += (c ^ a & (b ^ c)) + blocks[5] + 1200080426;
  2579. d = (d << 12 | d >>> 20) + a << 0;
  2580. c += (b ^ d & (a ^ b)) + blocks[6] - 1473231341;
  2581. c = (c << 17 | c >>> 15) + d << 0;
  2582. b += (a ^ c & (d ^ a)) + blocks[7] - 45705983;
  2583. b = (b << 22 | b >>> 10) + c << 0;
  2584. a += (d ^ b & (c ^ d)) + blocks[8] + 1770035416;
  2585. a = (a << 7 | a >>> 25) + b << 0;
  2586. d += (c ^ a & (b ^ c)) + blocks[9] - 1958414417;
  2587. d = (d << 12 | d >>> 20) + a << 0;
  2588. c += (b ^ d & (a ^ b)) + blocks[10] - 42063;
  2589. c = (c << 17 | c >>> 15) + d << 0;
  2590. b += (a ^ c & (d ^ a)) + blocks[11] - 1990404162;
  2591. b = (b << 22 | b >>> 10) + c << 0;
  2592. a += (d ^ b & (c ^ d)) + blocks[12] + 1804603682;
  2593. a = (a << 7 | a >>> 25) + b << 0;
  2594. d += (c ^ a & (b ^ c)) + blocks[13] - 40341101;
  2595. d = (d << 12 | d >>> 20) + a << 0;
  2596. c += (b ^ d & (a ^ b)) + blocks[14] - 1502002290;
  2597. c = (c << 17 | c >>> 15) + d << 0;
  2598. b += (a ^ c & (d ^ a)) + blocks[15] + 1236535329;
  2599. b = (b << 22 | b >>> 10) + c << 0;
  2600. a += (c ^ d & (b ^ c)) + blocks[1] - 165796510;
  2601. a = (a << 5 | a >>> 27) + b << 0;
  2602. d += (b ^ c & (a ^ b)) + blocks[6] - 1069501632;
  2603. d = (d << 9 | d >>> 23) + a << 0;
  2604. c += (a ^ b & (d ^ a)) + blocks[11] + 643717713;
  2605. c = (c << 14 | c >>> 18) + d << 0;
  2606. b += (d ^ a & (c ^ d)) + blocks[0] - 373897302;
  2607. b = (b << 20 | b >>> 12) + c << 0;
  2608. a += (c ^ d & (b ^ c)) + blocks[5] - 701558691;
  2609. a = (a << 5 | a >>> 27) + b << 0;
  2610. d += (b ^ c & (a ^ b)) + blocks[10] + 38016083;
  2611. d = (d << 9 | d >>> 23) + a << 0;
  2612. c += (a ^ b & (d ^ a)) + blocks[15] - 660478335;
  2613. c = (c << 14 | c >>> 18) + d << 0;
  2614. b += (d ^ a & (c ^ d)) + blocks[4] - 405537848;
  2615. b = (b << 20 | b >>> 12) + c << 0;
  2616. a += (c ^ d & (b ^ c)) + blocks[9] + 568446438;
  2617. a = (a << 5 | a >>> 27) + b << 0;
  2618. d += (b ^ c & (a ^ b)) + blocks[14] - 1019803690;
  2619. d = (d << 9 | d >>> 23) + a << 0;
  2620. c += (a ^ b & (d ^ a)) + blocks[3] - 187363961;
  2621. c = (c << 14 | c >>> 18) + d << 0;
  2622. b += (d ^ a & (c ^ d)) + blocks[8] + 1163531501;
  2623. b = (b << 20 | b >>> 12) + c << 0;
  2624. a += (c ^ d & (b ^ c)) + blocks[13] - 1444681467;
  2625. a = (a << 5 | a >>> 27) + b << 0;
  2626. d += (b ^ c & (a ^ b)) + blocks[2] - 51403784;
  2627. d = (d << 9 | d >>> 23) + a << 0;
  2628. c += (a ^ b & (d ^ a)) + blocks[7] + 1735328473;
  2629. c = (c << 14 | c >>> 18) + d << 0;
  2630. b += (d ^ a & (c ^ d)) + blocks[12] - 1926607734;
  2631. b = (b << 20 | b >>> 12) + c << 0;
  2632. bc = b ^ c;
  2633. a += (bc ^ d) + blocks[5] - 378558;
  2634. a = (a << 4 | a >>> 28) + b << 0;
  2635. d += (bc ^ a) + blocks[8] - 2022574463;
  2636. d = (d << 11 | d >>> 21) + a << 0;
  2637. da = d ^ a;
  2638. c += (da ^ b) + blocks[11] + 1839030562;
  2639. c = (c << 16 | c >>> 16) + d << 0;
  2640. b += (da ^ c) + blocks[14] - 35309556;
  2641. b = (b << 23 | b >>> 9) + c << 0;
  2642. bc = b ^ c;
  2643. a += (bc ^ d) + blocks[1] - 1530992060;
  2644. a = (a << 4 | a >>> 28) + b << 0;
  2645. d += (bc ^ a) + blocks[4] + 1272893353;
  2646. d = (d << 11 | d >>> 21) + a << 0;
  2647. da = d ^ a;
  2648. c += (da ^ b) + blocks[7] - 155497632;
  2649. c = (c << 16 | c >>> 16) + d << 0;
  2650. b += (da ^ c) + blocks[10] - 1094730640;
  2651. b = (b << 23 | b >>> 9) + c << 0;
  2652. bc = b ^ c;
  2653. a += (bc ^ d) + blocks[13] + 681279174;
  2654. a = (a << 4 | a >>> 28) + b << 0;
  2655. d += (bc ^ a) + blocks[0] - 358537222;
  2656. d = (d << 11 | d >>> 21) + a << 0;
  2657. da = d ^ a;
  2658. c += (da ^ b) + blocks[3] - 722521979;
  2659. c = (c << 16 | c >>> 16) + d << 0;
  2660. b += (da ^ c) + blocks[6] + 76029189;
  2661. b = (b << 23 | b >>> 9) + c << 0;
  2662. bc = b ^ c;
  2663. a += (bc ^ d) + blocks[9] - 640364487;
  2664. a = (a << 4 | a >>> 28) + b << 0;
  2665. d += (bc ^ a) + blocks[12] - 421815835;
  2666. d = (d << 11 | d >>> 21) + a << 0;
  2667. da = d ^ a;
  2668. c += (da ^ b) + blocks[15] + 530742520;
  2669. c = (c << 16 | c >>> 16) + d << 0;
  2670. b += (da ^ c) + blocks[2] - 995338651;
  2671. b = (b << 23 | b >>> 9) + c << 0;
  2672. a += (c ^ (b | ~d)) + blocks[0] - 198630844;
  2673. a = (a << 6 | a >>> 26) + b << 0;
  2674. d += (b ^ (a | ~c)) + blocks[7] + 1126891415;
  2675. d = (d << 10 | d >>> 22) + a << 0;
  2676. c += (a ^ (d | ~b)) + blocks[14] - 1416354905;
  2677. c = (c << 15 | c >>> 17) + d << 0;
  2678. b += (d ^ (c | ~a)) + blocks[5] - 57434055;
  2679. b = (b << 21 | b >>> 11) + c << 0;
  2680. a += (c ^ (b | ~d)) + blocks[12] + 1700485571;
  2681. a = (a << 6 | a >>> 26) + b << 0;
  2682. d += (b ^ (a | ~c)) + blocks[3] - 1894986606;
  2683. d = (d << 10 | d >>> 22) + a << 0;
  2684. c += (a ^ (d | ~b)) + blocks[10] - 1051523;
  2685. c = (c << 15 | c >>> 17) + d << 0;
  2686. b += (d ^ (c | ~a)) + blocks[1] - 2054922799;
  2687. b = (b << 21 | b >>> 11) + c << 0;
  2688. a += (c ^ (b | ~d)) + blocks[8] + 1873313359;
  2689. a = (a << 6 | a >>> 26) + b << 0;
  2690. d += (b ^ (a | ~c)) + blocks[15] - 30611744;
  2691. d = (d << 10 | d >>> 22) + a << 0;
  2692. c += (a ^ (d | ~b)) + blocks[6] - 1560198380;
  2693. c = (c << 15 | c >>> 17) + d << 0;
  2694. b += (d ^ (c | ~a)) + blocks[13] + 1309151649;
  2695. b = (b << 21 | b >>> 11) + c << 0;
  2696. a += (c ^ (b | ~d)) + blocks[4] - 145523070;
  2697. a = (a << 6 | a >>> 26) + b << 0;
  2698. d += (b ^ (a | ~c)) + blocks[11] - 1120210379;
  2699. d = (d << 10 | d >>> 22) + a << 0;
  2700. c += (a ^ (d | ~b)) + blocks[2] + 718787259;
  2701. c = (c << 15 | c >>> 17) + d << 0;
  2702. b += (d ^ (c | ~a)) + blocks[9] - 343485551;
  2703. b = (b << 21 | b >>> 11) + c << 0;
  2704. if (this.first) {
  2705. this.h0 = a + 1732584193 << 0;
  2706. this.h1 = b - 271733879 << 0;
  2707. this.h2 = c - 1732584194 << 0;
  2708. this.h3 = d + 271733878 << 0;
  2709. this.first = false;
  2710. } else {
  2711. this.h0 = this.h0 + a << 0;
  2712. this.h1 = this.h1 + b << 0;
  2713. this.h2 = this.h2 + c << 0;
  2714. this.h3 = this.h3 + d << 0;
  2715. }
  2716. };
  2717. /**
  2718. * @method hex
  2719. * @memberof Md5
  2720. * @instance
  2721. * @description Output hash as hex string
  2722. * @returns {String} Hex string
  2723. * @see {@link md5.hex}
  2724. * @example
  2725. * hash.hex();
  2726. */
  2727. Md5.prototype.hex = function () {
  2728. this.finalize();
  2729. var h0 = this.h0,
  2730. h1 = this.h1,
  2731. h2 = this.h2,
  2732. h3 = this.h3;
  2733. return HEX_CHARS[h0 >> 4 & 0x0F] + HEX_CHARS[h0 & 0x0F] + HEX_CHARS[h0 >> 12 & 0x0F] + HEX_CHARS[h0 >> 8 & 0x0F] + HEX_CHARS[h0 >> 20 & 0x0F] + HEX_CHARS[h0 >> 16 & 0x0F] + HEX_CHARS[h0 >> 28 & 0x0F] + HEX_CHARS[h0 >> 24 & 0x0F] + HEX_CHARS[h1 >> 4 & 0x0F] + HEX_CHARS[h1 & 0x0F] + HEX_CHARS[h1 >> 12 & 0x0F] + HEX_CHARS[h1 >> 8 & 0x0F] + HEX_CHARS[h1 >> 20 & 0x0F] + HEX_CHARS[h1 >> 16 & 0x0F] + HEX_CHARS[h1 >> 28 & 0x0F] + HEX_CHARS[h1 >> 24 & 0x0F] + HEX_CHARS[h2 >> 4 & 0x0F] + HEX_CHARS[h2 & 0x0F] + HEX_CHARS[h2 >> 12 & 0x0F] + HEX_CHARS[h2 >> 8 & 0x0F] + HEX_CHARS[h2 >> 20 & 0x0F] + HEX_CHARS[h2 >> 16 & 0x0F] + HEX_CHARS[h2 >> 28 & 0x0F] + HEX_CHARS[h2 >> 24 & 0x0F] + HEX_CHARS[h3 >> 4 & 0x0F] + HEX_CHARS[h3 & 0x0F] + HEX_CHARS[h3 >> 12 & 0x0F] + HEX_CHARS[h3 >> 8 & 0x0F] + HEX_CHARS[h3 >> 20 & 0x0F] + HEX_CHARS[h3 >> 16 & 0x0F] + HEX_CHARS[h3 >> 28 & 0x0F] + HEX_CHARS[h3 >> 24 & 0x0F];
  2734. };
  2735. /**
  2736. * @method toString
  2737. * @memberof Md5
  2738. * @instance
  2739. * @description Output hash as hex string
  2740. * @returns {String} Hex string
  2741. * @see {@link md5.hex}
  2742. * @example
  2743. * hash.toString();
  2744. */
  2745. Md5.prototype.toString = Md5.prototype.hex;
  2746. /**
  2747. * @method digest
  2748. * @memberof Md5
  2749. * @instance
  2750. * @description Output hash as bytes array
  2751. * @returns {Array} Bytes array
  2752. * @see {@link md5.digest}
  2753. * @example
  2754. * hash.digest();
  2755. */
  2756. Md5.prototype.digest = function (format) {
  2757. if (format === 'hex') return this.hex();
  2758. this.finalize();
  2759. var h0 = this.h0,
  2760. h1 = this.h1,
  2761. h2 = this.h2,
  2762. h3 = this.h3;
  2763. var res = [h0 & 0xFF, h0 >> 8 & 0xFF, h0 >> 16 & 0xFF, h0 >> 24 & 0xFF, h1 & 0xFF, h1 >> 8 & 0xFF, h1 >> 16 & 0xFF, h1 >> 24 & 0xFF, h2 & 0xFF, h2 >> 8 & 0xFF, h2 >> 16 & 0xFF, h2 >> 24 & 0xFF, h3 & 0xFF, h3 >> 8 & 0xFF, h3 >> 16 & 0xFF, h3 >> 24 & 0xFF];
  2764. return res;
  2765. };
  2766. /**
  2767. * @method array
  2768. * @memberof Md5
  2769. * @instance
  2770. * @description Output hash as bytes array
  2771. * @returns {Array} Bytes array
  2772. * @see {@link md5.array}
  2773. * @example
  2774. * hash.array();
  2775. */
  2776. Md5.prototype.array = Md5.prototype.digest;
  2777. /**
  2778. * @method arrayBuffer
  2779. * @memberof Md5
  2780. * @instance
  2781. * @description Output hash as ArrayBuffer
  2782. * @returns {ArrayBuffer} ArrayBuffer
  2783. * @see {@link md5.arrayBuffer}
  2784. * @example
  2785. * hash.arrayBuffer();
  2786. */
  2787. Md5.prototype.arrayBuffer = function () {
  2788. this.finalize();
  2789. var buffer = new ArrayBuffer(16);
  2790. var blocks = new Uint32Array(buffer);
  2791. blocks[0] = this.h0;
  2792. blocks[1] = this.h1;
  2793. blocks[2] = this.h2;
  2794. blocks[3] = this.h3;
  2795. return buffer;
  2796. };
  2797. /**
  2798. * @method buffer
  2799. * @deprecated This maybe confuse with Buffer in node.js. Please use arrayBuffer instead.
  2800. * @memberof Md5
  2801. * @instance
  2802. * @description Output hash as ArrayBuffer
  2803. * @returns {ArrayBuffer} ArrayBuffer
  2804. * @see {@link md5.buffer}
  2805. * @example
  2806. * hash.buffer();
  2807. */
  2808. Md5.prototype.buffer = Md5.prototype.arrayBuffer;
  2809. /**
  2810. * @method base64
  2811. * @memberof Md5
  2812. * @instance
  2813. * @description Output hash as base64 string
  2814. * @returns {String} base64 string
  2815. * @see {@link md5.base64}
  2816. * @example
  2817. * hash.base64();
  2818. */
  2819. Md5.prototype.base64 = function () {
  2820. var v1,
  2821. v2,
  2822. v3,
  2823. base64Str = '',
  2824. bytes = this.array();
  2825. for (var i = 0; i < 15;) {
  2826. v1 = bytes[i++];
  2827. v2 = bytes[i++];
  2828. v3 = bytes[i++];
  2829. base64Str += BASE64_ENCODE_CHAR[v1 >>> 2] + BASE64_ENCODE_CHAR[(v1 << 4 | v2 >>> 4) & 63] + BASE64_ENCODE_CHAR[(v2 << 2 | v3 >>> 6) & 63] + BASE64_ENCODE_CHAR[v3 & 63];
  2830. }
  2831. v1 = bytes[i];
  2832. base64Str += BASE64_ENCODE_CHAR[v1 >>> 2] + BASE64_ENCODE_CHAR[v1 << 4 & 63] + '==';
  2833. return base64Str;
  2834. };
  2835. var exports = createMethod();
  2836. if (COMMON_JS) {
  2837. module.exports = exports;
  2838. } else {
  2839. /**
  2840. * @method md5
  2841. * @description Md5 hash function, export to global in browsers.
  2842. * @param {String|Array|Uint8Array|ArrayBuffer} message message to hash
  2843. * @returns {String} md5 hashes
  2844. * @example
  2845. * md5(''); // d41d8cd98f00b204e9800998ecf8427e
  2846. * md5('The quick brown fox jumps over the lazy dog'); // 9e107d9d372bb6826bd81d3542a419d6
  2847. * md5('The quick brown fox jumps over the lazy dog.'); // e4d909c290d0fb1ca068ffaddf22cbd0
  2848. *
  2849. * // It also supports UTF-8 encoding
  2850. * md5('中文'); // a7bac2239fcdcb3a067903d8077c4a07
  2851. *
  2852. * // It also supports byte `Array`, `Uint8Array`, `ArrayBuffer`
  2853. * md5([]); // d41d8cd98f00b204e9800998ecf8427e
  2854. * md5(new Uint8Array([])); // d41d8cd98f00b204e9800998ecf8427e
  2855. */
  2856. root.md5 = exports;
  2857. if (AMD) {
  2858. !(__WEBPACK_AMD_DEFINE_RESULT__ = (function () {
  2859. return exports;
  2860. }).call(exports, __webpack_require__, exports, module),
  2861. __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__));
  2862. }
  2863. }
  2864. })();
  2865. /* WEBPACK VAR INJECTION */}.call(exports, __webpack_require__(1), __webpack_require__(8)))
  2866. /***/ }),
  2867. /* 8 */
  2868. /***/ (function(module, exports) {
  2869. var g;
  2870. // This works in non-strict mode
  2871. g = (function() {
  2872. return this;
  2873. })();
  2874. try {
  2875. // This works if eval is allowed (see CSP)
  2876. g = g || Function("return this")() || (1,eval)("this");
  2877. } catch(e) {
  2878. // This works if the window reference is available
  2879. if(typeof window === "object")
  2880. g = window;
  2881. }
  2882. // g can still be undefined, but nothing to do about it...
  2883. // We return undefined, instead of nothing here, so it's
  2884. // easier to handle this case. if(!global) { ...}
  2885. module.exports = g;
  2886. /***/ }),
  2887. /* 9 */
  2888. /***/ (function(module, exports) {
  2889. /* WEBPACK VAR INJECTION */(function(__webpack_amd_options__) {/* globals __webpack_amd_options__ */
  2890. module.exports = __webpack_amd_options__;
  2891. /* WEBPACK VAR INJECTION */}.call(exports, {}))
  2892. /***/ }),
  2893. /* 10 */
  2894. /***/ (function(module, exports, __webpack_require__) {
  2895. /*
  2896. CryptoJS v3.1.2
  2897. code.google.com/p/crypto-js
  2898. (c) 2009-2013 by Jeff Mott. All rights reserved.
  2899. code.google.com/p/crypto-js/wiki/License
  2900. */
  2901. var CryptoJS = CryptoJS || function (g, l) {
  2902. var e = {},
  2903. d = e.lib = {},
  2904. m = function () {},
  2905. k = d.Base = { extend: function (a) {
  2906. m.prototype = this;var c = new m();a && c.mixIn(a);c.hasOwnProperty("init") || (c.init = function () {
  2907. c.$super.init.apply(this, arguments);
  2908. });c.init.prototype = c;c.$super = this;return c;
  2909. }, create: function () {
  2910. var a = this.extend();a.init.apply(a, arguments);return a;
  2911. }, init: function () {}, mixIn: function (a) {
  2912. for (var c in a) a.hasOwnProperty(c) && (this[c] = a[c]);a.hasOwnProperty("toString") && (this.toString = a.toString);
  2913. }, clone: function () {
  2914. return this.init.prototype.extend(this);
  2915. } },
  2916. p = d.WordArray = k.extend({ init: function (a, c) {
  2917. a = this.words = a || [];this.sigBytes = c != l ? c : 4 * a.length;
  2918. }, toString: function (a) {
  2919. return (a || n).stringify(this);
  2920. }, concat: function (a) {
  2921. var c = this.words,
  2922. q = a.words,
  2923. f = this.sigBytes;a = a.sigBytes;this.clamp();if (f % 4) for (var b = 0; b < a; b++) c[f + b >>> 2] |= (q[b >>> 2] >>> 24 - 8 * (b % 4) & 255) << 24 - 8 * ((f + b) % 4);else if (65535 < q.length) for (b = 0; b < a; b += 4) c[f + b >>> 2] = q[b >>> 2];else c.push.apply(c, q);this.sigBytes += a;return this;
  2924. }, clamp: function () {
  2925. var a = this.words,
  2926. c = this.sigBytes;a[c >>> 2] &= 4294967295 << 32 - 8 * (c % 4);a.length = g.ceil(c / 4);
  2927. }, clone: function () {
  2928. var a = k.clone.call(this);a.words = this.words.slice(0);return a;
  2929. }, random: function (a) {
  2930. for (var c = [], b = 0; b < a; b += 4) c.push(4294967296 * g.random() | 0);return new p.init(c, a);
  2931. } }),
  2932. b = e.enc = {},
  2933. n = b.Hex = { stringify: function (a) {
  2934. var c = a.words;a = a.sigBytes;for (var b = [], f = 0; f < a; f++) {
  2935. var d = c[f >>> 2] >>> 24 - 8 * (f % 4) & 255;b.push((d >>> 4).toString(16));b.push((d & 15).toString(16));
  2936. }return b.join("");
  2937. }, parse: function (a) {
  2938. for (var c = a.length, b = [], f = 0; f < c; f += 2) b[f >>> 3] |= parseInt(a.substr(f, 2), 16) << 24 - 4 * (f % 8);return new p.init(b, c / 2);
  2939. } },
  2940. j = b.Latin1 = { stringify: function (a) {
  2941. var c = a.words;a = a.sigBytes;for (var b = [], f = 0; f < a; f++) b.push(String.fromCharCode(c[f >>> 2] >>> 24 - 8 * (f % 4) & 255));return b.join("");
  2942. }, parse: function (a) {
  2943. for (var c = a.length, b = [], f = 0; f < c; f++) b[f >>> 2] |= (a.charCodeAt(f) & 255) << 24 - 8 * (f % 4);return new p.init(b, c);
  2944. } },
  2945. h = b.Utf8 = { stringify: function (a) {
  2946. try {
  2947. return decodeURIComponent(escape(j.stringify(a)));
  2948. } catch (c) {
  2949. throw Error("Malformed UTF-8 data");
  2950. }
  2951. }, parse: function (a) {
  2952. return j.parse(unescape(encodeURIComponent(a)));
  2953. } },
  2954. r = d.BufferedBlockAlgorithm = k.extend({ reset: function () {
  2955. this._data = new p.init();this._nDataBytes = 0;
  2956. }, _append: function (a) {
  2957. "string" == typeof a && (a = h.parse(a));this._data.concat(a);this._nDataBytes += a.sigBytes;
  2958. }, _process: function (a) {
  2959. var c = this._data,
  2960. b = c.words,
  2961. f = c.sigBytes,
  2962. d = this.blockSize,
  2963. e = f / (4 * d),
  2964. e = a ? g.ceil(e) : g.max((e | 0) - this._minBufferSize, 0);a = e * d;f = g.min(4 * a, f);if (a) {
  2965. for (var k = 0; k < a; k += d) this._doProcessBlock(b, k);k = b.splice(0, a);c.sigBytes -= f;
  2966. }return new p.init(k, f);
  2967. }, clone: function () {
  2968. var a = k.clone.call(this);
  2969. a._data = this._data.clone();return a;
  2970. }, _minBufferSize: 0 });d.Hasher = r.extend({ cfg: k.extend(), init: function (a) {
  2971. this.cfg = this.cfg.extend(a);this.reset();
  2972. }, reset: function () {
  2973. r.reset.call(this);this._doReset();
  2974. }, update: function (a) {
  2975. this._append(a);this._process();return this;
  2976. }, finalize: function (a) {
  2977. a && this._append(a);return this._doFinalize();
  2978. }, blockSize: 16, _createHelper: function (a) {
  2979. return function (b, d) {
  2980. return new a.init(d).finalize(b);
  2981. };
  2982. }, _createHmacHelper: function (a) {
  2983. return function (b, d) {
  2984. return new s.HMAC.init(a, d).finalize(b);
  2985. };
  2986. } });var s = e.algo = {};return e;
  2987. }(Math);
  2988. (function () {
  2989. var g = CryptoJS,
  2990. l = g.lib,
  2991. e = l.WordArray,
  2992. d = l.Hasher,
  2993. m = [],
  2994. l = g.algo.SHA1 = d.extend({ _doReset: function () {
  2995. this._hash = new e.init([1732584193, 4023233417, 2562383102, 271733878, 3285377520]);
  2996. }, _doProcessBlock: function (d, e) {
  2997. for (var b = this._hash.words, n = b[0], j = b[1], h = b[2], g = b[3], l = b[4], a = 0; 80 > a; a++) {
  2998. if (16 > a) m[a] = d[e + a] | 0;else {
  2999. var c = m[a - 3] ^ m[a - 8] ^ m[a - 14] ^ m[a - 16];m[a] = c << 1 | c >>> 31;
  3000. }c = (n << 5 | n >>> 27) + l + m[a];c = 20 > a ? c + ((j & h | ~j & g) + 1518500249) : 40 > a ? c + ((j ^ h ^ g) + 1859775393) : 60 > a ? c + ((j & h | j & g | h & g) - 1894007588) : c + ((j ^ h ^ g) - 899497514);l = g;g = h;h = j << 30 | j >>> 2;j = n;n = c;
  3001. }b[0] = b[0] + n | 0;b[1] = b[1] + j | 0;b[2] = b[2] + h | 0;b[3] = b[3] + g | 0;b[4] = b[4] + l | 0;
  3002. }, _doFinalize: function () {
  3003. var d = this._data,
  3004. e = d.words,
  3005. b = 8 * this._nDataBytes,
  3006. g = 8 * d.sigBytes;e[g >>> 5] |= 128 << 24 - g % 32;e[(g + 64 >>> 9 << 4) + 14] = Math.floor(b / 4294967296);e[(g + 64 >>> 9 << 4) + 15] = b;d.sigBytes = 4 * e.length;this._process();return this._hash;
  3007. }, clone: function () {
  3008. var e = d.clone.call(this);e._hash = this._hash.clone();return e;
  3009. } });g.SHA1 = d._createHelper(l);g.HmacSHA1 = d._createHmacHelper(l);
  3010. })();
  3011. (function () {
  3012. var g = CryptoJS,
  3013. l = g.enc.Utf8;g.algo.HMAC = g.lib.Base.extend({ init: function (e, d) {
  3014. e = this._hasher = new e.init();"string" == typeof d && (d = l.parse(d));var g = e.blockSize,
  3015. k = 4 * g;d.sigBytes > k && (d = e.finalize(d));d.clamp();for (var p = this._oKey = d.clone(), b = this._iKey = d.clone(), n = p.words, j = b.words, h = 0; h < g; h++) n[h] ^= 1549556828, j[h] ^= 909522486;p.sigBytes = b.sigBytes = k;this.reset();
  3016. }, reset: function () {
  3017. var e = this._hasher;e.reset();e.update(this._iKey);
  3018. }, update: function (e) {
  3019. this._hasher.update(e);return this;
  3020. }, finalize: function (e) {
  3021. var d = this._hasher;e = d.finalize(e);d.reset();return d.finalize(this._oKey.clone().concat(e));
  3022. } });
  3023. })();
  3024. (function () {
  3025. // Shortcuts
  3026. var C = CryptoJS;
  3027. var C_lib = C.lib;
  3028. var WordArray = C_lib.WordArray;
  3029. var C_enc = C.enc;
  3030. /**
  3031. * Base64 encoding strategy.
  3032. */
  3033. var Base64 = C_enc.Base64 = {
  3034. /**
  3035. * Converts a word array to a Base64 string.
  3036. *
  3037. * @param {WordArray} wordArray The word array.
  3038. *
  3039. * @return {string} The Base64 string.
  3040. *
  3041. * @static
  3042. *
  3043. * @example
  3044. *
  3045. * var base64String = CryptoJS.enc.Base64.stringify(wordArray);
  3046. */
  3047. stringify: function (wordArray) {
  3048. // Shortcuts
  3049. var words = wordArray.words;
  3050. var sigBytes = wordArray.sigBytes;
  3051. var map = this._map;
  3052. // Clamp excess bits
  3053. wordArray.clamp();
  3054. // Convert
  3055. var base64Chars = [];
  3056. for (var i = 0; i < sigBytes; i += 3) {
  3057. var byte1 = words[i >>> 2] >>> 24 - i % 4 * 8 & 0xff;
  3058. var byte2 = words[i + 1 >>> 2] >>> 24 - (i + 1) % 4 * 8 & 0xff;
  3059. var byte3 = words[i + 2 >>> 2] >>> 24 - (i + 2) % 4 * 8 & 0xff;
  3060. var triplet = byte1 << 16 | byte2 << 8 | byte3;
  3061. for (var j = 0; j < 4 && i + j * 0.75 < sigBytes; j++) {
  3062. base64Chars.push(map.charAt(triplet >>> 6 * (3 - j) & 0x3f));
  3063. }
  3064. }
  3065. // Add padding
  3066. var paddingChar = map.charAt(64);
  3067. if (paddingChar) {
  3068. while (base64Chars.length % 4) {
  3069. base64Chars.push(paddingChar);
  3070. }
  3071. }
  3072. return base64Chars.join('');
  3073. },
  3074. /**
  3075. * Converts a Base64 string to a word array.
  3076. *
  3077. * @param {string} base64Str The Base64 string.
  3078. *
  3079. * @return {WordArray} The word array.
  3080. *
  3081. * @static
  3082. *
  3083. * @example
  3084. *
  3085. * var wordArray = CryptoJS.enc.Base64.parse(base64String);
  3086. */
  3087. parse: function (base64Str) {
  3088. // Shortcuts
  3089. var base64StrLength = base64Str.length;
  3090. var map = this._map;
  3091. // Ignore padding
  3092. var paddingChar = map.charAt(64);
  3093. if (paddingChar) {
  3094. var paddingIndex = base64Str.indexOf(paddingChar);
  3095. if (paddingIndex != -1) {
  3096. base64StrLength = paddingIndex;
  3097. }
  3098. }
  3099. // Convert
  3100. var words = [];
  3101. var nBytes = 0;
  3102. for (var i = 0; i < base64StrLength; i++) {
  3103. if (i % 4) {
  3104. var bits1 = map.indexOf(base64Str.charAt(i - 1)) << i % 4 * 2;
  3105. var bits2 = map.indexOf(base64Str.charAt(i)) >>> 6 - i % 4 * 2;
  3106. words[nBytes >>> 2] |= (bits1 | bits2) << 24 - nBytes % 4 * 8;
  3107. nBytes++;
  3108. }
  3109. }
  3110. return WordArray.create(words, nBytes);
  3111. },
  3112. _map: 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='
  3113. };
  3114. })();
  3115. if (true) {
  3116. module.exports = CryptoJS;
  3117. } else {
  3118. window.CryptoJS = CryptoJS;
  3119. }
  3120. /***/ }),
  3121. /* 11 */
  3122. /***/ (function(module, exports, __webpack_require__) {
  3123. /* Copyright 2015 William Summers, MetaTribal LLC
  3124. * adapted from https://developer.mozilla.org/en-US/docs/JXON
  3125. *
  3126. * Licensed under the MIT License, Version 2.0 (the "License");
  3127. * you may not use this file except in compliance with the License.
  3128. * You may obtain a copy of the License at
  3129. *
  3130. * https://opensource.org/licenses/MIT
  3131. *
  3132. * Unless required by applicable law or agreed to in writing, software
  3133. * distributed under the License is distributed on an "AS IS" BASIS,
  3134. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  3135. * See the License for the specific language governing permissions and
  3136. * limitations under the License.
  3137. */
  3138. /**
  3139. * @author William Summers
  3140. * https://github.com/metatribal/xmlToJSON
  3141. */
  3142. var DOMParser = __webpack_require__(12).DOMParser;
  3143. var xmlToJSON = function () {
  3144. this.version = "1.3.5";
  3145. var options = { // set up the default options
  3146. mergeCDATA: true, // extract cdata and merge with text
  3147. normalize: true, // collapse multiple spaces to single space
  3148. stripElemPrefix: true // for elements of same name in diff namespaces, you can enable namespaces and access the nskey property
  3149. };
  3150. var prefixMatch = new RegExp(/(?!xmlns)^.*:/);
  3151. var trimMatch = new RegExp(/^\s+|\s+$/g);
  3152. this.grokType = function (sValue) {
  3153. if (/^\s*$/.test(sValue)) {
  3154. return null;
  3155. }
  3156. if (/^(?:true|false)$/i.test(sValue)) {
  3157. return sValue.toLowerCase() === "true";
  3158. }
  3159. if (isFinite(sValue)) {
  3160. return parseFloat(sValue);
  3161. }
  3162. return sValue;
  3163. };
  3164. this.parseString = function (xmlString, opt) {
  3165. if (xmlString) {
  3166. var xml = this.stringToXML(xmlString);
  3167. if (xml.getElementsByTagName('parsererror').length) {
  3168. return null;
  3169. } else {
  3170. return this.parseXML(xml, opt);
  3171. }
  3172. } else {
  3173. return null;
  3174. }
  3175. };
  3176. this.parseXML = function (oXMLParent, opt) {
  3177. // initialize options
  3178. for (var key in opt) {
  3179. options[key] = opt[key];
  3180. }
  3181. var vResult = {},
  3182. nLength = 0,
  3183. sCollectedTxt = "";
  3184. // iterate over the children
  3185. var childNum = oXMLParent.childNodes.length;
  3186. if (childNum) {
  3187. for (var oNode, sProp, vContent, nItem = 0; nItem < oXMLParent.childNodes.length; nItem++) {
  3188. oNode = oXMLParent.childNodes.item(nItem);
  3189. if (oNode.nodeType === 4) {
  3190. if (options.mergeCDATA) {
  3191. sCollectedTxt += oNode.nodeValue;
  3192. }
  3193. } /* nodeType is "CDATASection" (4) */
  3194. else if (oNode.nodeType === 3) {
  3195. sCollectedTxt += oNode.nodeValue;
  3196. } /* nodeType is "Text" (3) */
  3197. else if (oNode.nodeType === 1) {
  3198. /* nodeType is "Element" (1) */
  3199. if (nLength === 0) {
  3200. vResult = {};
  3201. }
  3202. // using nodeName to support browser (IE) implementation with no 'localName' property
  3203. if (options.stripElemPrefix) {
  3204. sProp = oNode.nodeName.replace(prefixMatch, '');
  3205. } else {
  3206. sProp = oNode.nodeName;
  3207. }
  3208. vContent = xmlToJSON.parseXML(oNode);
  3209. if (vResult.hasOwnProperty(sProp)) {
  3210. if (vResult[sProp].constructor !== Array) {
  3211. vResult[sProp] = [vResult[sProp]];
  3212. }
  3213. vResult[sProp].push(vContent);
  3214. } else {
  3215. vResult[sProp] = vContent;
  3216. nLength++;
  3217. }
  3218. }
  3219. }
  3220. }
  3221. if (!Object.keys(vResult).length) {
  3222. // vResult = sCollectedTxt.replace(trimMatch, '') || ''; // by carsonxu 修复 getBucket返回的 Key 是 " /" 这种场景
  3223. vResult = sCollectedTxt || '';
  3224. }
  3225. return vResult;
  3226. };
  3227. // Convert xmlDocument to a string
  3228. // Returns null on failure
  3229. this.xmlToString = function (xmlDoc) {
  3230. try {
  3231. var xmlString = xmlDoc.xml ? xmlDoc.xml : new XMLSerializer().serializeToString(xmlDoc);
  3232. return xmlString;
  3233. } catch (err) {
  3234. return null;
  3235. }
  3236. };
  3237. // Convert a string to XML Node Structure
  3238. // Returns null on failure
  3239. this.stringToXML = function (xmlString) {
  3240. try {
  3241. var xmlDoc = null;
  3242. if (window.DOMParser) {
  3243. var parser = new DOMParser();
  3244. xmlDoc = parser.parseFromString(xmlString, "text/xml");
  3245. return xmlDoc;
  3246. } else {
  3247. xmlDoc = new ActiveXObject("Microsoft.XMLDOM");
  3248. xmlDoc.async = false;
  3249. xmlDoc.loadXML(xmlString);
  3250. return xmlDoc;
  3251. }
  3252. } catch (e) {
  3253. return null;
  3254. }
  3255. };
  3256. return this;
  3257. }.call({});
  3258. var xml2json = function (xmlString) {
  3259. return xmlToJSON.parseString(xmlString);
  3260. };
  3261. module.exports = xml2json;
  3262. /***/ }),
  3263. /* 12 */
  3264. /***/ (function(module, exports, __webpack_require__) {
  3265. function DOMParser(options){
  3266. this.options = options ||{locator:{}};
  3267. }
  3268. DOMParser.prototype.parseFromString = function(source,mimeType){
  3269. var options = this.options;
  3270. var sax = new XMLReader();
  3271. var domBuilder = options.domBuilder || new DOMHandler();//contentHandler and LexicalHandler
  3272. var errorHandler = options.errorHandler;
  3273. var locator = options.locator;
  3274. var defaultNSMap = options.xmlns||{};
  3275. var entityMap = {'lt':'<','gt':'>','amp':'&','quot':'"','apos':"'"}
  3276. if(locator){
  3277. domBuilder.setDocumentLocator(locator)
  3278. }
  3279. sax.errorHandler = buildErrorHandler(errorHandler,domBuilder,locator);
  3280. sax.domBuilder = options.domBuilder || domBuilder;
  3281. if(/\/x?html?$/.test(mimeType)){
  3282. entityMap.nbsp = '\xa0';
  3283. entityMap.copy = '\xa9';
  3284. defaultNSMap['']= 'http://www.w3.org/1999/xhtml';
  3285. }
  3286. defaultNSMap.xml = defaultNSMap.xml || 'http://www.w3.org/XML/1998/namespace';
  3287. if(source){
  3288. sax.parse(source,defaultNSMap,entityMap);
  3289. }else{
  3290. sax.errorHandler.error("invalid doc source");
  3291. }
  3292. return domBuilder.doc;
  3293. }
  3294. function buildErrorHandler(errorImpl,domBuilder,locator){
  3295. if(!errorImpl){
  3296. if(domBuilder instanceof DOMHandler){
  3297. return domBuilder;
  3298. }
  3299. errorImpl = domBuilder ;
  3300. }
  3301. var errorHandler = {}
  3302. var isCallback = errorImpl instanceof Function;
  3303. locator = locator||{}
  3304. function build(key){
  3305. var fn = errorImpl[key];
  3306. if(!fn && isCallback){
  3307. fn = errorImpl.length == 2?function(msg){errorImpl(key,msg)}:errorImpl;
  3308. }
  3309. errorHandler[key] = fn && function(msg){
  3310. fn('[xmldom '+key+']\t'+msg+_locator(locator));
  3311. }||function(){};
  3312. }
  3313. build('warning');
  3314. build('error');
  3315. build('fatalError');
  3316. return errorHandler;
  3317. }
  3318. //console.log('#\n\n\n\n\n\n\n####')
  3319. /**
  3320. * +ContentHandler+ErrorHandler
  3321. * +LexicalHandler+EntityResolver2
  3322. * -DeclHandler-DTDHandler
  3323. *
  3324. * DefaultHandler:EntityResolver, DTDHandler, ContentHandler, ErrorHandler
  3325. * DefaultHandler2:DefaultHandler,LexicalHandler, DeclHandler, EntityResolver2
  3326. * @link http://www.saxproject.org/apidoc/org/xml/sax/helpers/DefaultHandler.html
  3327. */
  3328. function DOMHandler() {
  3329. this.cdata = false;
  3330. }
  3331. function position(locator,node){
  3332. node.lineNumber = locator.lineNumber;
  3333. node.columnNumber = locator.columnNumber;
  3334. }
  3335. /**
  3336. * @see org.xml.sax.ContentHandler#startDocument
  3337. * @link http://www.saxproject.org/apidoc/org/xml/sax/ContentHandler.html
  3338. */
  3339. DOMHandler.prototype = {
  3340. startDocument : function() {
  3341. this.doc = new DOMImplementation().createDocument(null, null, null);
  3342. if (this.locator) {
  3343. this.doc.documentURI = this.locator.systemId;
  3344. }
  3345. },
  3346. startElement:function(namespaceURI, localName, qName, attrs) {
  3347. var doc = this.doc;
  3348. var el = doc.createElementNS(namespaceURI, qName||localName);
  3349. var len = attrs.length;
  3350. appendElement(this, el);
  3351. this.currentElement = el;
  3352. this.locator && position(this.locator,el)
  3353. for (var i = 0 ; i < len; i++) {
  3354. var namespaceURI = attrs.getURI(i);
  3355. var value = attrs.getValue(i);
  3356. var qName = attrs.getQName(i);
  3357. var attr = doc.createAttributeNS(namespaceURI, qName);
  3358. this.locator &&position(attrs.getLocator(i),attr);
  3359. attr.value = attr.nodeValue = value;
  3360. el.setAttributeNode(attr)
  3361. }
  3362. },
  3363. endElement:function(namespaceURI, localName, qName) {
  3364. var current = this.currentElement
  3365. var tagName = current.tagName;
  3366. this.currentElement = current.parentNode;
  3367. },
  3368. startPrefixMapping:function(prefix, uri) {
  3369. },
  3370. endPrefixMapping:function(prefix) {
  3371. },
  3372. processingInstruction:function(target, data) {
  3373. var ins = this.doc.createProcessingInstruction(target, data);
  3374. this.locator && position(this.locator,ins)
  3375. appendElement(this, ins);
  3376. },
  3377. ignorableWhitespace:function(ch, start, length) {
  3378. },
  3379. characters:function(chars, start, length) {
  3380. chars = _toString.apply(this,arguments)
  3381. //console.log(chars)
  3382. if(chars){
  3383. if (this.cdata) {
  3384. var charNode = this.doc.createCDATASection(chars);
  3385. } else {
  3386. var charNode = this.doc.createTextNode(chars);
  3387. }
  3388. if(this.currentElement){
  3389. this.currentElement.appendChild(charNode);
  3390. }else if(/^\s*$/.test(chars)){
  3391. this.doc.appendChild(charNode);
  3392. //process xml
  3393. }
  3394. this.locator && position(this.locator,charNode)
  3395. }
  3396. },
  3397. skippedEntity:function(name) {
  3398. },
  3399. endDocument:function() {
  3400. this.doc.normalize();
  3401. },
  3402. setDocumentLocator:function (locator) {
  3403. if(this.locator = locator){// && !('lineNumber' in locator)){
  3404. locator.lineNumber = 0;
  3405. }
  3406. },
  3407. //LexicalHandler
  3408. comment:function(chars, start, length) {
  3409. chars = _toString.apply(this,arguments)
  3410. var comm = this.doc.createComment(chars);
  3411. this.locator && position(this.locator,comm)
  3412. appendElement(this, comm);
  3413. },
  3414. startCDATA:function() {
  3415. //used in characters() methods
  3416. this.cdata = true;
  3417. },
  3418. endCDATA:function() {
  3419. this.cdata = false;
  3420. },
  3421. startDTD:function(name, publicId, systemId) {
  3422. var impl = this.doc.implementation;
  3423. if (impl && impl.createDocumentType) {
  3424. var dt = impl.createDocumentType(name, publicId, systemId);
  3425. this.locator && position(this.locator,dt)
  3426. appendElement(this, dt);
  3427. }
  3428. },
  3429. /**
  3430. * @see org.xml.sax.ErrorHandler
  3431. * @link http://www.saxproject.org/apidoc/org/xml/sax/ErrorHandler.html
  3432. */
  3433. warning:function(error) {
  3434. console.warn('[xmldom warning]\t'+error,_locator(this.locator));
  3435. },
  3436. error:function(error) {
  3437. console.error('[xmldom error]\t'+error,_locator(this.locator));
  3438. },
  3439. fatalError:function(error) {
  3440. console.error('[xmldom fatalError]\t'+error,_locator(this.locator));
  3441. throw error;
  3442. }
  3443. }
  3444. function _locator(l){
  3445. if(l){
  3446. return '\n@'+(l.systemId ||'')+'#[line:'+l.lineNumber+',col:'+l.columnNumber+']'
  3447. }
  3448. }
  3449. function _toString(chars,start,length){
  3450. if(typeof chars == 'string'){
  3451. return chars.substr(start,length)
  3452. }else{//java sax connect width xmldom on rhino(what about: "? && !(chars instanceof String)")
  3453. if(chars.length >= start+length || start){
  3454. return new java.lang.String(chars,start,length)+'';
  3455. }
  3456. return chars;
  3457. }
  3458. }
  3459. /*
  3460. * @link http://www.saxproject.org/apidoc/org/xml/sax/ext/LexicalHandler.html
  3461. * used method of org.xml.sax.ext.LexicalHandler:
  3462. * #comment(chars, start, length)
  3463. * #startCDATA()
  3464. * #endCDATA()
  3465. * #startDTD(name, publicId, systemId)
  3466. *
  3467. *
  3468. * IGNORED method of org.xml.sax.ext.LexicalHandler:
  3469. * #endDTD()
  3470. * #startEntity(name)
  3471. * #endEntity(name)
  3472. *
  3473. *
  3474. * @link http://www.saxproject.org/apidoc/org/xml/sax/ext/DeclHandler.html
  3475. * IGNORED method of org.xml.sax.ext.DeclHandler
  3476. * #attributeDecl(eName, aName, type, mode, value)
  3477. * #elementDecl(name, model)
  3478. * #externalEntityDecl(name, publicId, systemId)
  3479. * #internalEntityDecl(name, value)
  3480. * @link http://www.saxproject.org/apidoc/org/xml/sax/ext/EntityResolver2.html
  3481. * IGNORED method of org.xml.sax.EntityResolver2
  3482. * #resolveEntity(String name,String publicId,String baseURI,String systemId)
  3483. * #resolveEntity(publicId, systemId)
  3484. * #getExternalSubset(name, baseURI)
  3485. * @link http://www.saxproject.org/apidoc/org/xml/sax/DTDHandler.html
  3486. * IGNORED method of org.xml.sax.DTDHandler
  3487. * #notationDecl(name, publicId, systemId) {};
  3488. * #unparsedEntityDecl(name, publicId, systemId, notationName) {};
  3489. */
  3490. "endDTD,startEntity,endEntity,attributeDecl,elementDecl,externalEntityDecl,internalEntityDecl,resolveEntity,getExternalSubset,notationDecl,unparsedEntityDecl".replace(/\w+/g,function(key){
  3491. DOMHandler.prototype[key] = function(){return null}
  3492. })
  3493. /* Private static helpers treated below as private instance methods, so don't need to add these to the public API; we might use a Relator to also get rid of non-standard public properties */
  3494. function appendElement (hander,node) {
  3495. if (!hander.currentElement) {
  3496. hander.doc.appendChild(node);
  3497. } else {
  3498. hander.currentElement.appendChild(node);
  3499. }
  3500. }//appendChild and setAttributeNS are preformance key
  3501. //if(typeof require == 'function'){
  3502. var XMLReader = __webpack_require__(13).XMLReader;
  3503. var DOMImplementation = exports.DOMImplementation = __webpack_require__(2).DOMImplementation;
  3504. exports.XMLSerializer = __webpack_require__(2).XMLSerializer ;
  3505. exports.DOMParser = DOMParser;
  3506. //}
  3507. /***/ }),
  3508. /* 13 */
  3509. /***/ (function(module, exports) {
  3510. //[4] NameStartChar ::= ":" | [A-Z] | "_" | [a-z] | [#xC0-#xD6] | [#xD8-#xF6] | [#xF8-#x2FF] | [#x370-#x37D] | [#x37F-#x1FFF] | [#x200C-#x200D] | [#x2070-#x218F] | [#x2C00-#x2FEF] | [#x3001-#xD7FF] | [#xF900-#xFDCF] | [#xFDF0-#xFFFD] | [#x10000-#xEFFFF]
  3511. //[4a] NameChar ::= NameStartChar | "-" | "." | [0-9] | #xB7 | [#x0300-#x036F] | [#x203F-#x2040]
  3512. //[5] Name ::= NameStartChar (NameChar)*
  3513. var nameStartChar = /[A-Z_a-z\xC0-\xD6\xD8-\xF6\u00F8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD]///\u10000-\uEFFFF
  3514. var nameChar = new RegExp("[\\-\\.0-9"+nameStartChar.source.slice(1,-1)+"\\u00B7\\u0300-\\u036F\\u203F-\\u2040]");
  3515. var tagNamePattern = new RegExp('^'+nameStartChar.source+nameChar.source+'*(?:\:'+nameStartChar.source+nameChar.source+'*)?$');
  3516. //var tagNamePattern = /^[a-zA-Z_][\w\-\.]*(?:\:[a-zA-Z_][\w\-\.]*)?$/
  3517. //var handlers = 'resolveEntity,getExternalSubset,characters,endDocument,endElement,endPrefixMapping,ignorableWhitespace,processingInstruction,setDocumentLocator,skippedEntity,startDocument,startElement,startPrefixMapping,notationDecl,unparsedEntityDecl,error,fatalError,warning,attributeDecl,elementDecl,externalEntityDecl,internalEntityDecl,comment,endCDATA,endDTD,endEntity,startCDATA,startDTD,startEntity'.split(',')
  3518. //S_TAG, S_ATTR, S_EQ, S_ATTR_NOQUOT_VALUE
  3519. //S_ATTR_SPACE, S_ATTR_END, S_TAG_SPACE, S_TAG_CLOSE
  3520. var S_TAG = 0;//tag name offerring
  3521. var S_ATTR = 1;//attr name offerring
  3522. var S_ATTR_SPACE=2;//attr name end and space offer
  3523. var S_EQ = 3;//=space?
  3524. var S_ATTR_NOQUOT_VALUE = 4;//attr value(no quot value only)
  3525. var S_ATTR_END = 5;//attr value end and no space(quot end)
  3526. var S_TAG_SPACE = 6;//(attr value end || tag end ) && (space offer)
  3527. var S_TAG_CLOSE = 7;//closed el<el />
  3528. function XMLReader(){
  3529. }
  3530. XMLReader.prototype = {
  3531. parse:function(source,defaultNSMap,entityMap){
  3532. var domBuilder = this.domBuilder;
  3533. domBuilder.startDocument();
  3534. _copy(defaultNSMap ,defaultNSMap = {})
  3535. parse(source,defaultNSMap,entityMap,
  3536. domBuilder,this.errorHandler);
  3537. domBuilder.endDocument();
  3538. }
  3539. }
  3540. function parse(source,defaultNSMapCopy,entityMap,domBuilder,errorHandler){
  3541. function fixedFromCharCode(code) {
  3542. // String.prototype.fromCharCode does not supports
  3543. // > 2 bytes unicode chars directly
  3544. if (code > 0xffff) {
  3545. code -= 0x10000;
  3546. var surrogate1 = 0xd800 + (code >> 10)
  3547. , surrogate2 = 0xdc00 + (code & 0x3ff);
  3548. return String.fromCharCode(surrogate1, surrogate2);
  3549. } else {
  3550. return String.fromCharCode(code);
  3551. }
  3552. }
  3553. function entityReplacer(a){
  3554. var k = a.slice(1,-1);
  3555. if(k in entityMap){
  3556. return entityMap[k];
  3557. }else if(k.charAt(0) === '#'){
  3558. return fixedFromCharCode(parseInt(k.substr(1).replace('x','0x')))
  3559. }else{
  3560. errorHandler.error('entity not found:'+a);
  3561. return a;
  3562. }
  3563. }
  3564. function appendText(end){//has some bugs
  3565. if(end>start){
  3566. var xt = source.substring(start,end).replace(/&#?\w+;/g,entityReplacer);
  3567. locator&&position(start);
  3568. domBuilder.characters(xt,0,end-start);
  3569. start = end
  3570. }
  3571. }
  3572. function position(p,m){
  3573. while(p>=lineEnd && (m = linePattern.exec(source))){
  3574. lineStart = m.index;
  3575. lineEnd = lineStart + m[0].length;
  3576. locator.lineNumber++;
  3577. //console.log('line++:',locator,startPos,endPos)
  3578. }
  3579. locator.columnNumber = p-lineStart+1;
  3580. }
  3581. var lineStart = 0;
  3582. var lineEnd = 0;
  3583. var linePattern = /.*(?:\r\n?|\n)|.*$/g
  3584. var locator = domBuilder.locator;
  3585. var parseStack = [{currentNSMap:defaultNSMapCopy}]
  3586. var closeMap = {};
  3587. var start = 0;
  3588. while(true){
  3589. try{
  3590. var tagStart = source.indexOf('<',start);
  3591. if(tagStart<0){
  3592. if(!source.substr(start).match(/^\s*$/)){
  3593. var doc = domBuilder.doc;
  3594. var text = doc.createTextNode(source.substr(start));
  3595. doc.appendChild(text);
  3596. domBuilder.currentElement = text;
  3597. }
  3598. return;
  3599. }
  3600. if(tagStart>start){
  3601. appendText(tagStart);
  3602. }
  3603. switch(source.charAt(tagStart+1)){
  3604. case '/':
  3605. var end = source.indexOf('>',tagStart+3);
  3606. var tagName = source.substring(tagStart+2,end);
  3607. var config = parseStack.pop();
  3608. if(end<0){
  3609. tagName = source.substring(tagStart+2).replace(/[\s<].*/,'');
  3610. //console.error('#@@@@@@'+tagName)
  3611. errorHandler.error("end tag name: "+tagName+' is not complete:'+config.tagName);
  3612. end = tagStart+1+tagName.length;
  3613. }else if(tagName.match(/\s</)){
  3614. tagName = tagName.replace(/[\s<].*/,'');
  3615. errorHandler.error("end tag name: "+tagName+' maybe not complete');
  3616. end = tagStart+1+tagName.length;
  3617. }
  3618. //console.error(parseStack.length,parseStack)
  3619. //console.error(config);
  3620. var localNSMap = config.localNSMap;
  3621. var endMatch = config.tagName == tagName;
  3622. var endIgnoreCaseMach = endMatch || config.tagName&&config.tagName.toLowerCase() == tagName.toLowerCase()
  3623. if(endIgnoreCaseMach){
  3624. domBuilder.endElement(config.uri,config.localName,tagName);
  3625. if(localNSMap){
  3626. for(var prefix in localNSMap){
  3627. domBuilder.endPrefixMapping(prefix) ;
  3628. }
  3629. }
  3630. if(!endMatch){
  3631. errorHandler.fatalError("end tag name: "+tagName+' is not match the current start tagName:'+config.tagName );
  3632. }
  3633. }else{
  3634. parseStack.push(config)
  3635. }
  3636. end++;
  3637. break;
  3638. // end elment
  3639. case '?':// <?...?>
  3640. locator&&position(tagStart);
  3641. end = parseInstruction(source,tagStart,domBuilder);
  3642. break;
  3643. case '!':// <!doctype,<![CDATA,<!--
  3644. locator&&position(tagStart);
  3645. end = parseDCC(source,tagStart,domBuilder,errorHandler);
  3646. break;
  3647. default:
  3648. locator&&position(tagStart);
  3649. var el = new ElementAttributes();
  3650. var currentNSMap = parseStack[parseStack.length-1].currentNSMap;
  3651. //elStartEnd
  3652. var end = parseElementStartPart(source,tagStart,el,currentNSMap,entityReplacer,errorHandler);
  3653. var len = el.length;
  3654. if(!el.closed && fixSelfClosed(source,end,el.tagName,closeMap)){
  3655. el.closed = true;
  3656. if(!entityMap.nbsp){
  3657. errorHandler.warning('unclosed xml attribute');
  3658. }
  3659. }
  3660. if(locator && len){
  3661. var locator2 = copyLocator(locator,{});
  3662. //try{//attribute position fixed
  3663. for(var i = 0;i<len;i++){
  3664. var a = el[i];
  3665. position(a.offset);
  3666. a.locator = copyLocator(locator,{});
  3667. }
  3668. //}catch(e){console.error('@@@@@'+e)}
  3669. domBuilder.locator = locator2
  3670. if(appendElement(el,domBuilder,currentNSMap)){
  3671. parseStack.push(el)
  3672. }
  3673. domBuilder.locator = locator;
  3674. }else{
  3675. if(appendElement(el,domBuilder,currentNSMap)){
  3676. parseStack.push(el)
  3677. }
  3678. }
  3679. if(el.uri === 'http://www.w3.org/1999/xhtml' && !el.closed){
  3680. end = parseHtmlSpecialContent(source,end,el.tagName,entityReplacer,domBuilder)
  3681. }else{
  3682. end++;
  3683. }
  3684. }
  3685. }catch(e){
  3686. errorHandler.error('element parse error: '+e)
  3687. //errorHandler.error('element parse error: '+e);
  3688. end = -1;
  3689. //throw e;
  3690. }
  3691. if(end>start){
  3692. start = end;
  3693. }else{
  3694. //TODO: 这里有可能sax回退,有位置错误风险
  3695. appendText(Math.max(tagStart,start)+1);
  3696. }
  3697. }
  3698. }
  3699. function copyLocator(f,t){
  3700. t.lineNumber = f.lineNumber;
  3701. t.columnNumber = f.columnNumber;
  3702. return t;
  3703. }
  3704. /**
  3705. * @see #appendElement(source,elStartEnd,el,selfClosed,entityReplacer,domBuilder,parseStack);
  3706. * @return end of the elementStartPart(end of elementEndPart for selfClosed el)
  3707. */
  3708. function parseElementStartPart(source,start,el,currentNSMap,entityReplacer,errorHandler){
  3709. var attrName;
  3710. var value;
  3711. var p = ++start;
  3712. var s = S_TAG;//status
  3713. while(true){
  3714. var c = source.charAt(p);
  3715. switch(c){
  3716. case '=':
  3717. if(s === S_ATTR){//attrName
  3718. attrName = source.slice(start,p);
  3719. s = S_EQ;
  3720. }else if(s === S_ATTR_SPACE){
  3721. s = S_EQ;
  3722. }else{
  3723. //fatalError: equal must after attrName or space after attrName
  3724. throw new Error('attribute equal must after attrName');
  3725. }
  3726. break;
  3727. case '\'':
  3728. case '"':
  3729. if(s === S_EQ || s === S_ATTR //|| s == S_ATTR_SPACE
  3730. ){//equal
  3731. if(s === S_ATTR){
  3732. errorHandler.warning('attribute value must after "="')
  3733. attrName = source.slice(start,p)
  3734. }
  3735. start = p+1;
  3736. p = source.indexOf(c,start)
  3737. if(p>0){
  3738. value = source.slice(start,p).replace(/&#?\w+;/g,entityReplacer);
  3739. el.add(attrName,value,start-1);
  3740. s = S_ATTR_END;
  3741. }else{
  3742. //fatalError: no end quot match
  3743. throw new Error('attribute value no end \''+c+'\' match');
  3744. }
  3745. }else if(s == S_ATTR_NOQUOT_VALUE){
  3746. value = source.slice(start,p).replace(/&#?\w+;/g,entityReplacer);
  3747. //console.log(attrName,value,start,p)
  3748. el.add(attrName,value,start);
  3749. //console.dir(el)
  3750. errorHandler.warning('attribute "'+attrName+'" missed start quot('+c+')!!');
  3751. start = p+1;
  3752. s = S_ATTR_END
  3753. }else{
  3754. //fatalError: no equal before
  3755. throw new Error('attribute value must after "="');
  3756. }
  3757. break;
  3758. case '/':
  3759. switch(s){
  3760. case S_TAG:
  3761. el.setTagName(source.slice(start,p));
  3762. case S_ATTR_END:
  3763. case S_TAG_SPACE:
  3764. case S_TAG_CLOSE:
  3765. s =S_TAG_CLOSE;
  3766. el.closed = true;
  3767. case S_ATTR_NOQUOT_VALUE:
  3768. case S_ATTR:
  3769. case S_ATTR_SPACE:
  3770. break;
  3771. //case S_EQ:
  3772. default:
  3773. throw new Error("attribute invalid close char('/')")
  3774. }
  3775. break;
  3776. case ''://end document
  3777. //throw new Error('unexpected end of input')
  3778. errorHandler.error('unexpected end of input');
  3779. if(s == S_TAG){
  3780. el.setTagName(source.slice(start,p));
  3781. }
  3782. return p;
  3783. case '>':
  3784. switch(s){
  3785. case S_TAG:
  3786. el.setTagName(source.slice(start,p));
  3787. case S_ATTR_END:
  3788. case S_TAG_SPACE:
  3789. case S_TAG_CLOSE:
  3790. break;//normal
  3791. case S_ATTR_NOQUOT_VALUE://Compatible state
  3792. case S_ATTR:
  3793. value = source.slice(start,p);
  3794. if(value.slice(-1) === '/'){
  3795. el.closed = true;
  3796. value = value.slice(0,-1)
  3797. }
  3798. case S_ATTR_SPACE:
  3799. if(s === S_ATTR_SPACE){
  3800. value = attrName;
  3801. }
  3802. if(s == S_ATTR_NOQUOT_VALUE){
  3803. errorHandler.warning('attribute "'+value+'" missed quot(")!!');
  3804. el.add(attrName,value.replace(/&#?\w+;/g,entityReplacer),start)
  3805. }else{
  3806. if(currentNSMap[''] !== 'http://www.w3.org/1999/xhtml' || !value.match(/^(?:disabled|checked|selected)$/i)){
  3807. errorHandler.warning('attribute "'+value+'" missed value!! "'+value+'" instead!!')
  3808. }
  3809. el.add(value,value,start)
  3810. }
  3811. break;
  3812. case S_EQ:
  3813. throw new Error('attribute value missed!!');
  3814. }
  3815. // console.log(tagName,tagNamePattern,tagNamePattern.test(tagName))
  3816. return p;
  3817. /*xml space '\x20' | #x9 | #xD | #xA; */
  3818. case '\u0080':
  3819. c = ' ';
  3820. default:
  3821. if(c<= ' '){//space
  3822. switch(s){
  3823. case S_TAG:
  3824. el.setTagName(source.slice(start,p));//tagName
  3825. s = S_TAG_SPACE;
  3826. break;
  3827. case S_ATTR:
  3828. attrName = source.slice(start,p)
  3829. s = S_ATTR_SPACE;
  3830. break;
  3831. case S_ATTR_NOQUOT_VALUE:
  3832. var value = source.slice(start,p).replace(/&#?\w+;/g,entityReplacer);
  3833. errorHandler.warning('attribute "'+value+'" missed quot(")!!');
  3834. el.add(attrName,value,start)
  3835. case S_ATTR_END:
  3836. s = S_TAG_SPACE;
  3837. break;
  3838. //case S_TAG_SPACE:
  3839. //case S_EQ:
  3840. //case S_ATTR_SPACE:
  3841. // void();break;
  3842. //case S_TAG_CLOSE:
  3843. //ignore warning
  3844. }
  3845. }else{//not space
  3846. //S_TAG, S_ATTR, S_EQ, S_ATTR_NOQUOT_VALUE
  3847. //S_ATTR_SPACE, S_ATTR_END, S_TAG_SPACE, S_TAG_CLOSE
  3848. switch(s){
  3849. //case S_TAG:void();break;
  3850. //case S_ATTR:void();break;
  3851. //case S_ATTR_NOQUOT_VALUE:void();break;
  3852. case S_ATTR_SPACE:
  3853. var tagName = el.tagName;
  3854. if(currentNSMap[''] !== 'http://www.w3.org/1999/xhtml' || !attrName.match(/^(?:disabled|checked|selected)$/i)){
  3855. errorHandler.warning('attribute "'+attrName+'" missed value!! "'+attrName+'" instead2!!')
  3856. }
  3857. el.add(attrName,attrName,start);
  3858. start = p;
  3859. s = S_ATTR;
  3860. break;
  3861. case S_ATTR_END:
  3862. errorHandler.warning('attribute space is required"'+attrName+'"!!')
  3863. case S_TAG_SPACE:
  3864. s = S_ATTR;
  3865. start = p;
  3866. break;
  3867. case S_EQ:
  3868. s = S_ATTR_NOQUOT_VALUE;
  3869. start = p;
  3870. break;
  3871. case S_TAG_CLOSE:
  3872. throw new Error("elements closed character '/' and '>' must be connected to");
  3873. }
  3874. }
  3875. }//end outer switch
  3876. //console.log('p++',p)
  3877. p++;
  3878. }
  3879. }
  3880. /**
  3881. * @return true if has new namespace define
  3882. */
  3883. function appendElement(el,domBuilder,currentNSMap){
  3884. var tagName = el.tagName;
  3885. var localNSMap = null;
  3886. //var currentNSMap = parseStack[parseStack.length-1].currentNSMap;
  3887. var i = el.length;
  3888. while(i--){
  3889. var a = el[i];
  3890. var qName = a.qName;
  3891. var value = a.value;
  3892. var nsp = qName.indexOf(':');
  3893. if(nsp>0){
  3894. var prefix = a.prefix = qName.slice(0,nsp);
  3895. var localName = qName.slice(nsp+1);
  3896. var nsPrefix = prefix === 'xmlns' && localName
  3897. }else{
  3898. localName = qName;
  3899. prefix = null
  3900. nsPrefix = qName === 'xmlns' && ''
  3901. }
  3902. //can not set prefix,because prefix !== ''
  3903. a.localName = localName ;
  3904. //prefix == null for no ns prefix attribute
  3905. if(nsPrefix !== false){//hack!!
  3906. if(localNSMap == null){
  3907. localNSMap = {}
  3908. //console.log(currentNSMap,0)
  3909. _copy(currentNSMap,currentNSMap={})
  3910. //console.log(currentNSMap,1)
  3911. }
  3912. currentNSMap[nsPrefix] = localNSMap[nsPrefix] = value;
  3913. a.uri = 'http://www.w3.org/2000/xmlns/'
  3914. domBuilder.startPrefixMapping(nsPrefix, value)
  3915. }
  3916. }
  3917. var i = el.length;
  3918. while(i--){
  3919. a = el[i];
  3920. var prefix = a.prefix;
  3921. if(prefix){//no prefix attribute has no namespace
  3922. if(prefix === 'xml'){
  3923. a.uri = 'http://www.w3.org/XML/1998/namespace';
  3924. }if(prefix !== 'xmlns'){
  3925. a.uri = currentNSMap[prefix || '']
  3926. //{console.log('###'+a.qName,domBuilder.locator.systemId+'',currentNSMap,a.uri)}
  3927. }
  3928. }
  3929. }
  3930. var nsp = tagName.indexOf(':');
  3931. if(nsp>0){
  3932. prefix = el.prefix = tagName.slice(0,nsp);
  3933. localName = el.localName = tagName.slice(nsp+1);
  3934. }else{
  3935. prefix = null;//important!!
  3936. localName = el.localName = tagName;
  3937. }
  3938. //no prefix element has default namespace
  3939. var ns = el.uri = currentNSMap[prefix || ''];
  3940. domBuilder.startElement(ns,localName,tagName,el);
  3941. //endPrefixMapping and startPrefixMapping have not any help for dom builder
  3942. //localNSMap = null
  3943. if(el.closed){
  3944. domBuilder.endElement(ns,localName,tagName);
  3945. if(localNSMap){
  3946. for(prefix in localNSMap){
  3947. domBuilder.endPrefixMapping(prefix)
  3948. }
  3949. }
  3950. }else{
  3951. el.currentNSMap = currentNSMap;
  3952. el.localNSMap = localNSMap;
  3953. //parseStack.push(el);
  3954. return true;
  3955. }
  3956. }
  3957. function parseHtmlSpecialContent(source,elStartEnd,tagName,entityReplacer,domBuilder){
  3958. if(/^(?:script|textarea)$/i.test(tagName)){
  3959. var elEndStart = source.indexOf('</'+tagName+'>',elStartEnd);
  3960. var text = source.substring(elStartEnd+1,elEndStart);
  3961. if(/[&<]/.test(text)){
  3962. if(/^script$/i.test(tagName)){
  3963. //if(!/\]\]>/.test(text)){
  3964. //lexHandler.startCDATA();
  3965. domBuilder.characters(text,0,text.length);
  3966. //lexHandler.endCDATA();
  3967. return elEndStart;
  3968. //}
  3969. }//}else{//text area
  3970. text = text.replace(/&#?\w+;/g,entityReplacer);
  3971. domBuilder.characters(text,0,text.length);
  3972. return elEndStart;
  3973. //}
  3974. }
  3975. }
  3976. return elStartEnd+1;
  3977. }
  3978. function fixSelfClosed(source,elStartEnd,tagName,closeMap){
  3979. //if(tagName in closeMap){
  3980. var pos = closeMap[tagName];
  3981. if(pos == null){
  3982. //console.log(tagName)
  3983. pos = source.lastIndexOf('</'+tagName+'>')
  3984. if(pos<elStartEnd){//忘记闭合
  3985. pos = source.lastIndexOf('</'+tagName)
  3986. }
  3987. closeMap[tagName] =pos
  3988. }
  3989. return pos<elStartEnd;
  3990. //}
  3991. }
  3992. function _copy(source,target){
  3993. for(var n in source){target[n] = source[n]}
  3994. }
  3995. function parseDCC(source,start,domBuilder,errorHandler){//sure start with '<!'
  3996. var next= source.charAt(start+2)
  3997. switch(next){
  3998. case '-':
  3999. if(source.charAt(start + 3) === '-'){
  4000. var end = source.indexOf('-->',start+4);
  4001. //append comment source.substring(4,end)//<!--
  4002. if(end>start){
  4003. domBuilder.comment(source,start+4,end-start-4);
  4004. return end+3;
  4005. }else{
  4006. errorHandler.error("Unclosed comment");
  4007. return -1;
  4008. }
  4009. }else{
  4010. //error
  4011. return -1;
  4012. }
  4013. default:
  4014. if(source.substr(start+3,6) == 'CDATA['){
  4015. var end = source.indexOf(']]>',start+9);
  4016. domBuilder.startCDATA();
  4017. domBuilder.characters(source,start+9,end-start-9);
  4018. domBuilder.endCDATA()
  4019. return end+3;
  4020. }
  4021. //<!DOCTYPE
  4022. //startDTD(java.lang.String name, java.lang.String publicId, java.lang.String systemId)
  4023. var matchs = split(source,start);
  4024. var len = matchs.length;
  4025. if(len>1 && /!doctype/i.test(matchs[0][0])){
  4026. var name = matchs[1][0];
  4027. var pubid = len>3 && /^public$/i.test(matchs[2][0]) && matchs[3][0]
  4028. var sysid = len>4 && matchs[4][0];
  4029. var lastMatch = matchs[len-1]
  4030. domBuilder.startDTD(name,pubid && pubid.replace(/^(['"])(.*?)\1$/,'$2'),
  4031. sysid && sysid.replace(/^(['"])(.*?)\1$/,'$2'));
  4032. domBuilder.endDTD();
  4033. return lastMatch.index+lastMatch[0].length
  4034. }
  4035. }
  4036. return -1;
  4037. }
  4038. function parseInstruction(source,start,domBuilder){
  4039. var end = source.indexOf('?>',start);
  4040. if(end){
  4041. var match = source.substring(start,end).match(/^<\?(\S*)\s*([\s\S]*?)\s*$/);
  4042. if(match){
  4043. var len = match[0].length;
  4044. domBuilder.processingInstruction(match[1], match[2]) ;
  4045. return end+2;
  4046. }else{//error
  4047. return -1;
  4048. }
  4049. }
  4050. return -1;
  4051. }
  4052. /**
  4053. * @param source
  4054. */
  4055. function ElementAttributes(source){
  4056. }
  4057. ElementAttributes.prototype = {
  4058. setTagName:function(tagName){
  4059. if(!tagNamePattern.test(tagName)){
  4060. throw new Error('invalid tagName:'+tagName)
  4061. }
  4062. this.tagName = tagName
  4063. },
  4064. add:function(qName,value,offset){
  4065. if(!tagNamePattern.test(qName)){
  4066. throw new Error('invalid attribute:'+qName)
  4067. }
  4068. this[this.length++] = {qName:qName,value:value,offset:offset}
  4069. },
  4070. length:0,
  4071. getLocalName:function(i){return this[i].localName},
  4072. getLocator:function(i){return this[i].locator},
  4073. getQName:function(i){return this[i].qName},
  4074. getURI:function(i){return this[i].uri},
  4075. getValue:function(i){return this[i].value}
  4076. // ,getIndex:function(uri, localName)){
  4077. // if(localName){
  4078. //
  4079. // }else{
  4080. // var qName = uri
  4081. // }
  4082. // },
  4083. // getValue:function(){return this.getValue(this.getIndex.apply(this,arguments))},
  4084. // getType:function(uri,localName){}
  4085. // getType:function(i){},
  4086. }
  4087. function _set_proto_(thiz,parent){
  4088. thiz.__proto__ = parent;
  4089. return thiz;
  4090. }
  4091. if(!(_set_proto_({},_set_proto_.prototype) instanceof _set_proto_)){
  4092. _set_proto_ = function(thiz,parent){
  4093. function p(){};
  4094. p.prototype = parent;
  4095. p = new p();
  4096. for(parent in thiz){
  4097. p[parent] = thiz[parent];
  4098. }
  4099. return p;
  4100. }
  4101. }
  4102. function split(source,start){
  4103. var match;
  4104. var buf = [];
  4105. var reg = /'[^']+'|"[^"]+"|[^\s<>\/=]+=?|(\/?\s*>|<)/g;
  4106. reg.lastIndex = start;
  4107. reg.exec(source);//skip <
  4108. while(match = reg.exec(source)){
  4109. buf.push(match);
  4110. if(match[1])return buf;
  4111. }
  4112. }
  4113. exports.XMLReader = XMLReader;
  4114. /***/ }),
  4115. /* 14 */
  4116. /***/ (function(module, exports) {
  4117. //copyright Ryan Day 2010 <http://ryanday.org>, Joscha Feth 2013 <http://www.feth.com> [MIT Licensed]
  4118. var element_start_char = "a-zA-Z_\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u00FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FFF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD";
  4119. var element_non_start_char = "\-.0-9\u00B7\u0300-\u036F\u203F\u2040";
  4120. var element_replace = new RegExp("^([^" + element_start_char + "])|^((x|X)(m|M)(l|L))|([^" + element_start_char + element_non_start_char + "])", "g");
  4121. var not_safe_in_xml = /[^\x09\x0A\x0D\x20-\xFF\x85\xA0-\uD7FF\uE000-\uFDCF\uFDE0-\uFFFD]/gm;
  4122. var objKeys = function (obj) {
  4123. var l = [];
  4124. if (obj instanceof Object) {
  4125. for (var k in obj) {
  4126. if (obj.hasOwnProperty(k)) {
  4127. l.push(k);
  4128. }
  4129. }
  4130. }
  4131. return l;
  4132. };
  4133. var process_to_xml = function (node_data, options) {
  4134. var makeNode = function (name, content, attributes, level, hasSubNodes) {
  4135. var indent_value = options.indent !== undefined ? options.indent : "\t";
  4136. var indent = options.prettyPrint ? '\n' + new Array(level).join(indent_value) : '';
  4137. if (options.removeIllegalNameCharacters) {
  4138. name = name.replace(element_replace, '_');
  4139. }
  4140. var node = [indent, '<', name, attributes || ''];
  4141. if (content && content.length > 0) {
  4142. node.push('>');
  4143. node.push(content);
  4144. hasSubNodes && node.push(indent);
  4145. node.push('</');
  4146. node.push(name);
  4147. node.push('>');
  4148. } else {
  4149. node.push('/>');
  4150. }
  4151. return node.join('');
  4152. };
  4153. return function fn(node_data, node_descriptor, level) {
  4154. var type = typeof node_data;
  4155. if (Array.isArray ? Array.isArray(node_data) : node_data instanceof Array) {
  4156. type = 'array';
  4157. } else if (node_data instanceof Date) {
  4158. type = 'date';
  4159. }
  4160. switch (type) {
  4161. //if value is an array create child nodes from values
  4162. case 'array':
  4163. var ret = [];
  4164. node_data.map(function (v) {
  4165. ret.push(fn(v, 1, level + 1));
  4166. //entries that are values of an array are the only ones that can be special node descriptors
  4167. });
  4168. options.prettyPrint && ret.push('\n');
  4169. return ret.join('');
  4170. break;
  4171. case 'date':
  4172. // cast dates to ISO 8601 date (soap likes it)
  4173. return node_data.toJSON ? node_data.toJSON() : node_data + '';
  4174. break;
  4175. case 'object':
  4176. var nodes = [];
  4177. for (var name in node_data) {
  4178. if (node_data.hasOwnProperty(name)) {
  4179. if (node_data[name] instanceof Array) {
  4180. for (var j = 0; j < node_data[name].length; j++) {
  4181. if (node_data[name].hasOwnProperty(j)) {
  4182. nodes.push(makeNode(name, fn(node_data[name][j], 0, level + 1), null, level + 1, objKeys(node_data[name][j]).length));
  4183. }
  4184. }
  4185. } else {
  4186. nodes.push(makeNode(name, fn(node_data[name], 0, level + 1), null, level + 1));
  4187. }
  4188. }
  4189. }
  4190. options.prettyPrint && nodes.length > 0 && nodes.push('\n');
  4191. return nodes.join('');
  4192. break;
  4193. case 'function':
  4194. return node_data();
  4195. break;
  4196. default:
  4197. return options.escape ? esc(node_data) : '' + node_data;
  4198. }
  4199. }(node_data, 0, 0);
  4200. };
  4201. var xml_header = function (standalone) {
  4202. var ret = ['<?xml version="1.0" encoding="UTF-8"'];
  4203. if (standalone) {
  4204. ret.push(' standalone="yes"');
  4205. }
  4206. ret.push('?>');
  4207. return ret.join('');
  4208. };
  4209. function esc(str) {
  4210. return ('' + str).replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/'/g, '&apos;').replace(/"/g, '&quot;').replace(not_safe_in_xml, '');
  4211. }
  4212. module.exports = function (obj, options) {
  4213. if (!options) {
  4214. options = {
  4215. xmlHeader: {
  4216. standalone: true
  4217. },
  4218. prettyPrint: true,
  4219. indent: " ",
  4220. escape: true
  4221. };
  4222. }
  4223. if (typeof obj == 'string') {
  4224. try {
  4225. obj = JSON.parse(obj.toString());
  4226. } catch (e) {
  4227. return false;
  4228. }
  4229. }
  4230. var xmlheader = '';
  4231. var docType = '';
  4232. if (options) {
  4233. if (typeof options == 'object') {
  4234. // our config is an object
  4235. if (options.xmlHeader) {
  4236. // the user wants an xml header
  4237. xmlheader = xml_header(!!options.xmlHeader.standalone);
  4238. }
  4239. if (typeof options.docType != 'undefined') {
  4240. docType = '<!DOCTYPE ' + options.docType + '>';
  4241. }
  4242. } else {
  4243. // our config is a boolean value, so just add xml header
  4244. xmlheader = xml_header();
  4245. }
  4246. }
  4247. options = options || {};
  4248. var ret = [xmlheader, options.prettyPrint && docType ? '\n' : '', docType, process_to_xml(obj, options)];
  4249. return ret.join('').replace(/\n{2,}/g, '\n').replace(/\s+$/g, '');
  4250. };
  4251. /***/ }),
  4252. /* 15 */
  4253. /***/ (function(module, exports, __webpack_require__) {
  4254. var session = __webpack_require__(4);
  4255. var util = __webpack_require__(0);
  4256. var originApiMap = {};
  4257. var transferToTaskMethod = function (apiMap, apiName) {
  4258. originApiMap[apiName] = apiMap[apiName];
  4259. apiMap[apiName] = function (params, callback) {
  4260. if (params.SkipTask) {
  4261. originApiMap[apiName].call(this, params, callback);
  4262. } else {
  4263. this._addTask(apiName, params, callback);
  4264. }
  4265. };
  4266. };
  4267. var initTask = function (cos) {
  4268. var queue = [];
  4269. var tasks = {};
  4270. var uploadingFileCount = 0;
  4271. var nextUploadIndex = 0;
  4272. // 接口返回简略的任务信息
  4273. var formatTask = function (task) {
  4274. var t = {
  4275. id: task.id,
  4276. Bucket: task.Bucket,
  4277. Region: task.Region,
  4278. Key: task.Key,
  4279. FilePath: task.FilePath,
  4280. state: task.state,
  4281. loaded: task.loaded,
  4282. size: task.size,
  4283. speed: task.speed,
  4284. percent: task.percent,
  4285. hashPercent: task.hashPercent,
  4286. error: task.error
  4287. };
  4288. if (task.FilePath) t.FilePath = task.FilePath;
  4289. if (task._custom) t._custom = task._custom; // 控制台使用
  4290. return t;
  4291. };
  4292. var emitListUpdate = function () {
  4293. var timer;
  4294. var emit = function () {
  4295. timer = 0;
  4296. cos.emit('task-list-update', { list: util.map(queue, formatTask) });
  4297. cos.emit('list-update', { list: util.map(queue, formatTask) });
  4298. };
  4299. return function () {
  4300. if (!timer) timer = setTimeout(emit);
  4301. };
  4302. }();
  4303. var clearQueue = function () {
  4304. if (queue.length <= cos.options.UploadQueueSize) return;
  4305. for (var i = 0; i < nextUploadIndex && // 小于当前操作的 index 才清理
  4306. i < queue.length && // 大于队列才清理
  4307. queue.length > cos.options.UploadQueueSize // 如果还太多,才继续清理
  4308. ;) {
  4309. var isActive = queue[i].state === 'waiting' || queue[i].state === 'checking' || queue[i].state === 'uploading';
  4310. if (!queue[i] || !isActive) {
  4311. tasks[queue[i].id] && delete tasks[queue[i].id];
  4312. queue.splice(i, 1);
  4313. nextUploadIndex--;
  4314. } else {
  4315. i++;
  4316. }
  4317. }
  4318. emitListUpdate();
  4319. };
  4320. var startNextTask = function () {
  4321. // 检查是否允许增加执行进程
  4322. if (uploadingFileCount >= cos.options.FileParallelLimit) return;
  4323. // 跳过不可执行的任务
  4324. while (queue[nextUploadIndex] && queue[nextUploadIndex].state !== 'waiting') nextUploadIndex++;
  4325. // 检查是否已遍历结束
  4326. if (nextUploadIndex >= queue.length) return;
  4327. // 上传该遍历到的任务
  4328. var task = queue[nextUploadIndex];
  4329. nextUploadIndex++;
  4330. uploadingFileCount++;
  4331. task.state = 'checking';
  4332. task.params.onTaskStart && task.params.onTaskStart(formatTask(task));
  4333. !task.params.UploadData && (task.params.UploadData = {});
  4334. var apiParams = util.formatParams(task.api, task.params);
  4335. originApiMap[task.api].call(cos, apiParams, function (err, data) {
  4336. if (!cos._isRunningTask(task.id)) return;
  4337. if (task.state === 'checking' || task.state === 'uploading') {
  4338. task.state = err ? 'error' : 'success';
  4339. err && (task.error = err);
  4340. uploadingFileCount--;
  4341. emitListUpdate();
  4342. startNextTask();
  4343. task.callback && task.callback(err, data);
  4344. if (task.state === 'success') {
  4345. if (task.params) {
  4346. delete task.params.UploadData;
  4347. delete task.params.Body;
  4348. delete task.params;
  4349. }
  4350. delete task.callback;
  4351. }
  4352. }
  4353. clearQueue();
  4354. });
  4355. emitListUpdate();
  4356. // 异步执行下一个任务
  4357. setTimeout(startNextTask);
  4358. };
  4359. var killTask = function (id, switchToState) {
  4360. var task = tasks[id];
  4361. if (!task) return;
  4362. var waiting = task && task.state === 'waiting';
  4363. var running = task && (task.state === 'checking' || task.state === 'uploading');
  4364. if (switchToState === 'canceled' && task.state !== 'canceled' || switchToState === 'paused' && waiting || switchToState === 'paused' && running) {
  4365. if (switchToState === 'paused' && task.params.Body && typeof task.params.Body.pipe === 'function') {
  4366. console.error('stream not support pause');
  4367. return;
  4368. }
  4369. task.state = switchToState;
  4370. cos.emit('inner-kill-task', { TaskId: id, toState: switchToState });
  4371. try {
  4372. var UploadId = task && task.params && task.params.UploadData.UploadId;
  4373. } catch (e) {}
  4374. if (switchToState === 'canceled' && UploadId) session.removeUsing(UploadId);
  4375. emitListUpdate();
  4376. if (running) {
  4377. uploadingFileCount--;
  4378. startNextTask();
  4379. }
  4380. if (switchToState === 'canceled') {
  4381. if (task.params) {
  4382. delete task.params.UploadData;
  4383. delete task.params.Body;
  4384. delete task.params;
  4385. }
  4386. delete task.callback;
  4387. }
  4388. }
  4389. clearQueue();
  4390. };
  4391. cos._addTasks = function (taskList) {
  4392. util.each(taskList, function (task) {
  4393. cos._addTask(task.api, task.params, task.callback, true);
  4394. });
  4395. emitListUpdate();
  4396. };
  4397. var isTaskReadyWarning = true;
  4398. cos._addTask = function (api, params, callback, ignoreAddEvent) {
  4399. // 复制参数对象
  4400. params = util.formatParams(api, params);
  4401. // 生成 id
  4402. var id = util.uuid();
  4403. params.TaskId = id;
  4404. params.onTaskReady && params.onTaskReady(id);
  4405. if (params.TaskReady) {
  4406. params.TaskReady(id);
  4407. isTaskReadyWarning && console.warn('warning: Param "TaskReady" has been deprecated. Please use "onTaskReady" instead.');
  4408. isTaskReadyWarning = false;
  4409. }
  4410. var task = {
  4411. // env
  4412. params: params,
  4413. callback: callback,
  4414. api: api,
  4415. index: queue.length,
  4416. // task
  4417. id: id,
  4418. Bucket: params.Bucket,
  4419. Region: params.Region,
  4420. Key: params.Key,
  4421. FilePath: params.FilePath || '',
  4422. state: 'waiting',
  4423. loaded: 0,
  4424. size: 0,
  4425. speed: 0,
  4426. percent: 0,
  4427. hashPercent: 0,
  4428. error: null,
  4429. _custom: params._custom
  4430. };
  4431. var onHashProgress = params.onHashProgress;
  4432. params.onHashProgress = function (info) {
  4433. if (!cos._isRunningTask(task.id)) return;
  4434. task.hashPercent = info.percent;
  4435. onHashProgress && onHashProgress(info);
  4436. emitListUpdate();
  4437. };
  4438. var onProgress = params.onProgress;
  4439. params.onProgress = function (info) {
  4440. if (!cos._isRunningTask(task.id)) return;
  4441. task.state === 'checking' && (task.state = 'uploading');
  4442. task.loaded = info.loaded;
  4443. task.speed = info.speed;
  4444. task.percent = info.percent;
  4445. onProgress && onProgress(info);
  4446. emitListUpdate();
  4447. };
  4448. // 异步获取 filesize
  4449. util.getFileSize(api, params, function (err, size) {
  4450. // 开始处理上传
  4451. if (err) return callback(util.error(err)); // 如果获取大小出错,不加入队列
  4452. // 获取完文件大小再把任务加入队列
  4453. tasks[id] = task;
  4454. queue.push(task);
  4455. task.size = size;
  4456. !ignoreAddEvent && emitListUpdate();
  4457. startNextTask();
  4458. clearQueue();
  4459. });
  4460. return id;
  4461. };
  4462. cos._isRunningTask = function (id) {
  4463. var task = tasks[id];
  4464. return !!(task && (task.state === 'checking' || task.state === 'uploading'));
  4465. };
  4466. cos.getTaskList = function () {
  4467. return util.map(queue, formatTask);
  4468. };
  4469. cos.cancelTask = function (id) {
  4470. killTask(id, 'canceled');
  4471. };
  4472. cos.pauseTask = function (id) {
  4473. killTask(id, 'paused');
  4474. };
  4475. cos.restartTask = function (id) {
  4476. var task = tasks[id];
  4477. if (task && (task.state === 'paused' || task.state === 'error')) {
  4478. task.state = 'waiting';
  4479. emitListUpdate();
  4480. nextUploadIndex = Math.min(nextUploadIndex, task.index);
  4481. startNextTask();
  4482. }
  4483. };
  4484. cos.isUploadRunning = function () {
  4485. return uploadingFileCount || nextUploadIndex < queue.length;
  4486. };
  4487. };
  4488. module.exports.transferToTaskMethod = transferToTaskMethod;
  4489. module.exports.init = initTask;
  4490. /***/ }),
  4491. /* 16 */
  4492. /***/ (function(module, exports, __webpack_require__) {
  4493. var REQUEST = __webpack_require__(17);
  4494. var util = __webpack_require__(0);
  4495. // Bucket 相关
  4496. /**
  4497. * 获取用户的 bucket 列表
  4498. * @param {Object} params 回调函数,必须,下面为参数列表
  4499. * 无特殊参数
  4500. * @param {Function} callback 回调函数,必须
  4501. */
  4502. function getService(params, callback) {
  4503. if (typeof params === 'function') {
  4504. callback = params;
  4505. params = {};
  4506. }
  4507. var protocol = this.options.Protocol || (util.isBrowser && location.protocol === 'http:' ? 'http:' : 'https:');
  4508. var domain = this.options.ServiceDomain;
  4509. var appId = params.AppId || this.options.appId;
  4510. var region = params.Region;
  4511. if (domain) {
  4512. domain = domain.replace(/\{\{AppId\}\}/ig, appId || '').replace(/\{\{Region\}\}/ig, region || '').replace(/\{\{.*?\}\}/ig, '');
  4513. if (!/^[a-zA-Z]+:\/\//.test(domain)) {
  4514. domain = protocol + '//' + domain;
  4515. }
  4516. if (domain.slice(-1) === '/') {
  4517. domain = domain.slice(0, -1);
  4518. }
  4519. } else if (region) {
  4520. domain = protocol + '//cos.' + region + '.myqcloud.com';
  4521. } else {
  4522. domain = protocol + '//service.cos.myqcloud.com';
  4523. }
  4524. submitRequest.call(this, {
  4525. Action: 'name/cos:GetService',
  4526. url: domain,
  4527. method: 'GET',
  4528. headers: params.Headers
  4529. }, function (err, data) {
  4530. if (err) return callback(err);
  4531. var buckets = data && data.ListAllMyBucketsResult && data.ListAllMyBucketsResult.Buckets && data.ListAllMyBucketsResult.Buckets.Bucket || [];
  4532. buckets = util.isArray(buckets) ? buckets : [buckets];
  4533. var owner = data && data.ListAllMyBucketsResult && data.ListAllMyBucketsResult.Owner || {};
  4534. callback(null, {
  4535. Buckets: buckets,
  4536. Owner: owner,
  4537. statusCode: data.statusCode,
  4538. headers: data.headers
  4539. });
  4540. });
  4541. }
  4542. /**
  4543. * 创建 Bucket,并初始化访问权限
  4544. * @param {Object} params 参数对象,必须
  4545. * @param {String} params.Bucket Bucket名称,必须
  4546. * @param {String} params.Region 地域名称,必须
  4547. * @param {String} params.ACL 用户自定义文件权限,可以设置:private,public-read;默认值:private,非必须
  4548. * @param {String} params.GrantRead 赋予被授权者读的权限,格式x-cos-grant-read: uin=" ",uin=" ",非必须
  4549. * @param {String} params.GrantWrite 赋予被授权者写的权限,格式x-cos-grant-write: uin=" ",uin=" ",非必须
  4550. * @param {String} params.GrantFullControl 赋予被授权者读写权限,格式x-cos-grant-full-control: uin=" ",uin=" ",非必须
  4551. * @param {Function} callback 回调函数,必须
  4552. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  4553. * @return {Object} data 返回的数据
  4554. * @return {String} data.Location 操作地址
  4555. */
  4556. function putBucket(params, callback) {
  4557. var self = this;
  4558. var xml = '';
  4559. if (params['BucketAZConfig']) {
  4560. var CreateBucketConfiguration = {
  4561. BucketAZConfig: params.BucketAZConfig
  4562. };
  4563. xml = util.json2xml({ CreateBucketConfiguration: CreateBucketConfiguration });
  4564. }
  4565. submitRequest.call(this, {
  4566. Action: 'name/cos:PutBucket',
  4567. method: 'PUT',
  4568. Bucket: params.Bucket,
  4569. Region: params.Region,
  4570. headers: params.Headers,
  4571. body: xml
  4572. }, function (err, data) {
  4573. if (err) return callback(err);
  4574. var url = getUrl({
  4575. protocol: self.options.Protocol,
  4576. domain: self.options.Domain,
  4577. bucket: params.Bucket,
  4578. region: params.Region,
  4579. isLocation: true
  4580. });
  4581. callback(null, {
  4582. Location: url,
  4583. statusCode: data.statusCode,
  4584. headers: data.headers
  4585. });
  4586. });
  4587. }
  4588. /**
  4589. * 查看是否存在该Bucket,是否有权限访问
  4590. * @param {Object} params 参数对象,必须
  4591. * @param {String} params.Bucket Bucket名称,必须
  4592. * @param {String} params.Region 地域名称,必须
  4593. * @param {Function} callback 回调函数,必须
  4594. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  4595. * @return {Object} data 返回的数据
  4596. * @return {Boolean} data.BucketExist Bucket是否存在
  4597. * @return {Boolean} data.BucketAuth 是否有 Bucket 的访问权限
  4598. */
  4599. function headBucket(params, callback) {
  4600. submitRequest.call(this, {
  4601. Action: 'name/cos:HeadBucket',
  4602. Bucket: params.Bucket,
  4603. Region: params.Region,
  4604. headers: params.Headers,
  4605. method: 'HEAD'
  4606. }, callback);
  4607. }
  4608. /**
  4609. * 获取 Bucket 下的 object 列表
  4610. * @param {Object} params 参数对象,必须
  4611. * @param {String} params.Bucket Bucket名称,必须
  4612. * @param {String} params.Region 地域名称,必须
  4613. * @param {String} params.Prefix 前缀匹配,用来规定返回的文件前缀地址,非必须
  4614. * @param {String} params.Delimiter 定界符为一个符号,如果有Prefix,则将Prefix到delimiter之间的相同路径归为一类,非必须
  4615. * @param {String} params.Marker 默认以UTF-8二进制顺序列出条目,所有列出条目从marker开始,非必须
  4616. * @param {String} params.MaxKeys 单次返回最大的条目数量,默认1000,非必须
  4617. * @param {String} params.EncodingType 规定返回值的编码方式,非必须
  4618. * @param {Function} callback 回调函数,必须
  4619. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  4620. * @return {Object} data 返回的数据
  4621. * @return {Object} data.ListBucketResult 返回的 object 列表信息
  4622. */
  4623. function getBucket(params, callback) {
  4624. var reqParams = {};
  4625. reqParams['prefix'] = params['Prefix'] || '';
  4626. reqParams['delimiter'] = params['Delimiter'];
  4627. reqParams['marker'] = params['Marker'];
  4628. reqParams['max-keys'] = params['MaxKeys'];
  4629. reqParams['encoding-type'] = params['EncodingType'];
  4630. submitRequest.call(this, {
  4631. Action: 'name/cos:GetBucket',
  4632. ResourceKey: reqParams['prefix'],
  4633. method: 'GET',
  4634. Bucket: params.Bucket,
  4635. Region: params.Region,
  4636. headers: params.Headers,
  4637. qs: reqParams
  4638. }, function (err, data) {
  4639. if (err) return callback(err);
  4640. var ListBucketResult = data.ListBucketResult || {};
  4641. var Contents = ListBucketResult.Contents || [];
  4642. var CommonPrefixes = ListBucketResult.CommonPrefixes || [];
  4643. Contents = util.isArray(Contents) ? Contents : [Contents];
  4644. CommonPrefixes = util.isArray(CommonPrefixes) ? CommonPrefixes : [CommonPrefixes];
  4645. var result = util.clone(ListBucketResult);
  4646. util.extend(result, {
  4647. Contents: Contents,
  4648. CommonPrefixes: CommonPrefixes,
  4649. statusCode: data.statusCode,
  4650. headers: data.headers
  4651. });
  4652. callback(null, result);
  4653. });
  4654. }
  4655. /**
  4656. * 删除 Bucket
  4657. * @param {Object} params 参数对象,必须
  4658. * @param {String} params.Bucket Bucket名称,必须
  4659. * @param {String} params.Region 地域名称,必须
  4660. * @param {Function} callback 回调函数,必须
  4661. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  4662. * @return {Object} data 返回的数据
  4663. * @return {String} data.Location 操作地址
  4664. */
  4665. function deleteBucket(params, callback) {
  4666. submitRequest.call(this, {
  4667. Action: 'name/cos:DeleteBucket',
  4668. Bucket: params.Bucket,
  4669. Region: params.Region,
  4670. headers: params.Headers,
  4671. method: 'DELETE'
  4672. }, function (err, data) {
  4673. if (err && err.statusCode === 204) {
  4674. return callback(null, { statusCode: err.statusCode });
  4675. } else if (err) {
  4676. return callback(err);
  4677. }
  4678. callback(null, {
  4679. statusCode: data.statusCode,
  4680. headers: data.headers
  4681. });
  4682. });
  4683. }
  4684. /**
  4685. * 设置 Bucket 的 权限列表
  4686. * @param {Object} params 参数对象,必须
  4687. * @param {String} params.Bucket Bucket名称,必须
  4688. * @param {String} params.Region 地域名称,必须
  4689. * @param {String} params.ACL 用户自定义文件权限,可以设置:private,public-read;默认值:private,非必须
  4690. * @param {String} params.GrantRead 赋予被授权者读的权限,格式x-cos-grant-read: uin=" ",uin=" ",非必须
  4691. * @param {String} params.GrantWrite 赋予被授权者写的权限,格式x-cos-grant-write: uin=" ",uin=" ",非必须
  4692. * @param {String} params.GrantFullControl 赋予被授权者读写权限,格式x-cos-grant-full-control: uin=" ",uin=" ",非必须
  4693. * @param {Function} callback 回调函数,必须
  4694. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  4695. * @return {Object} data 返回的数据
  4696. */
  4697. function putBucketAcl(params, callback) {
  4698. var headers = params.Headers;
  4699. var xml = '';
  4700. if (params['AccessControlPolicy']) {
  4701. var AccessControlPolicy = util.clone(params['AccessControlPolicy'] || {});
  4702. var Grants = AccessControlPolicy.Grants || AccessControlPolicy.Grant;
  4703. Grants = util.isArray(Grants) ? Grants : [Grants];
  4704. delete AccessControlPolicy.Grant;
  4705. delete AccessControlPolicy.Grants;
  4706. AccessControlPolicy.AccessControlList = { Grant: Grants };
  4707. xml = util.json2xml({ AccessControlPolicy: AccessControlPolicy });
  4708. headers['Content-Type'] = 'application/xml';
  4709. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  4710. }
  4711. // Grant Header 去重
  4712. util.each(headers, function (val, key) {
  4713. if (key.indexOf('x-cos-grant-') === 0) {
  4714. headers[key] = uniqGrant(headers[key]);
  4715. }
  4716. });
  4717. submitRequest.call(this, {
  4718. Action: 'name/cos:PutBucketACL',
  4719. method: 'PUT',
  4720. Bucket: params.Bucket,
  4721. Region: params.Region,
  4722. headers: headers,
  4723. action: 'acl',
  4724. body: xml
  4725. }, function (err, data) {
  4726. if (err) return callback(err);
  4727. callback(null, {
  4728. statusCode: data.statusCode,
  4729. headers: data.headers
  4730. });
  4731. });
  4732. }
  4733. /**
  4734. * 获取 Bucket 的 权限列表
  4735. * @param {Object} params 参数对象,必须
  4736. * @param {String} params.Bucket Bucket名称,必须
  4737. * @param {String} params.Region 地域名称,必须
  4738. * @param {Function} callback 回调函数,必须
  4739. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  4740. * @return {Object} data 返回的数据
  4741. * @return {Object} data.AccessControlPolicy 访问权限信息
  4742. */
  4743. function getBucketAcl(params, callback) {
  4744. submitRequest.call(this, {
  4745. Action: 'name/cos:GetBucketACL',
  4746. method: 'GET',
  4747. Bucket: params.Bucket,
  4748. Region: params.Region,
  4749. headers: params.Headers,
  4750. action: 'acl'
  4751. }, function (err, data) {
  4752. if (err) return callback(err);
  4753. var AccessControlPolicy = data.AccessControlPolicy || {};
  4754. var Owner = AccessControlPolicy.Owner || {};
  4755. var Grant = AccessControlPolicy.AccessControlList.Grant || [];
  4756. Grant = util.isArray(Grant) ? Grant : [Grant];
  4757. var result = decodeAcl(AccessControlPolicy);
  4758. if (data.headers && data.headers['x-cos-acl']) {
  4759. result.ACL = data.headers['x-cos-acl'];
  4760. }
  4761. result = util.extend(result, {
  4762. Owner: Owner,
  4763. Grants: Grant,
  4764. statusCode: data.statusCode,
  4765. headers: data.headers
  4766. });
  4767. callback(null, result);
  4768. });
  4769. }
  4770. /**
  4771. * 设置 Bucket 的 跨域设置
  4772. * @param {Object} params 参数对象,必须
  4773. * @param {String} params.Bucket Bucket名称,必须
  4774. * @param {String} params.Region 地域名称,必须
  4775. * @param {Object} params.CORSConfiguration 相关的跨域设置,必须
  4776. * @param {Array} params.CORSConfiguration.CORSRules 对应的跨域规则
  4777. * @param {Function} callback 回调函数,必须
  4778. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  4779. * @return {Object} data 返回的数据
  4780. */
  4781. function putBucketCors(params, callback) {
  4782. var CORSConfiguration = params['CORSConfiguration'] || {};
  4783. var CORSRules = CORSConfiguration['CORSRules'] || params['CORSRules'] || [];
  4784. CORSRules = util.clone(util.isArray(CORSRules) ? CORSRules : [CORSRules]);
  4785. util.each(CORSRules, function (rule) {
  4786. util.each(['AllowedOrigin', 'AllowedHeader', 'AllowedMethod', 'ExposeHeader'], function (key) {
  4787. var sKey = key + 's';
  4788. var val = rule[sKey] || rule[key] || [];
  4789. delete rule[sKey];
  4790. rule[key] = util.isArray(val) ? val : [val];
  4791. });
  4792. });
  4793. var xml = util.json2xml({ CORSConfiguration: { CORSRule: CORSRules } });
  4794. var headers = params.Headers;
  4795. headers['Content-Type'] = 'application/xml';
  4796. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  4797. submitRequest.call(this, {
  4798. Action: 'name/cos:PutBucketCORS',
  4799. method: 'PUT',
  4800. Bucket: params.Bucket,
  4801. Region: params.Region,
  4802. body: xml,
  4803. action: 'cors',
  4804. headers: headers
  4805. }, function (err, data) {
  4806. if (err) return callback(err);
  4807. callback(null, {
  4808. statusCode: data.statusCode,
  4809. headers: data.headers
  4810. });
  4811. });
  4812. }
  4813. /**
  4814. * 获取 Bucket 的 跨域设置
  4815. * @param {Object} params 参数对象,必须
  4816. * @param {String} params.Bucket Bucket名称,必须
  4817. * @param {String} params.Region 地域名称,必须
  4818. * @param {Function} callback 回调函数,必须
  4819. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  4820. * @return {Object} data 返回的数据
  4821. * @return {Object} data.CORSRules Bucket的跨域设置
  4822. */
  4823. function getBucketCors(params, callback) {
  4824. submitRequest.call(this, {
  4825. Action: 'name/cos:GetBucketCORS',
  4826. method: 'GET',
  4827. Bucket: params.Bucket,
  4828. Region: params.Region,
  4829. headers: params.Headers,
  4830. action: 'cors'
  4831. }, function (err, data) {
  4832. if (err) {
  4833. if (err.statusCode === 404 && err.error && err.error.Code === 'NoSuchCORSConfiguration') {
  4834. var result = {
  4835. CORSRules: [],
  4836. statusCode: err.statusCode
  4837. };
  4838. err.headers && (result.headers = err.headers);
  4839. callback(null, result);
  4840. } else {
  4841. callback(err);
  4842. }
  4843. return;
  4844. }
  4845. var CORSConfiguration = data.CORSConfiguration || {};
  4846. var CORSRules = CORSConfiguration.CORSRules || CORSConfiguration.CORSRule || [];
  4847. CORSRules = util.clone(util.isArray(CORSRules) ? CORSRules : [CORSRules]);
  4848. util.each(CORSRules, function (rule) {
  4849. util.each(['AllowedOrigin', 'AllowedHeader', 'AllowedMethod', 'ExposeHeader'], function (key) {
  4850. var sKey = key + 's';
  4851. var val = rule[sKey] || rule[key] || [];
  4852. delete rule[key];
  4853. rule[sKey] = util.isArray(val) ? val : [val];
  4854. });
  4855. });
  4856. callback(null, {
  4857. CORSRules: CORSRules,
  4858. statusCode: data.statusCode,
  4859. headers: data.headers
  4860. });
  4861. });
  4862. }
  4863. /**
  4864. * 删除 Bucket 的 跨域设置
  4865. * @param {Object} params 参数对象,必须
  4866. * @param {String} params.Bucket Bucket名称,必须
  4867. * @param {String} params.Region 地域名称,必须
  4868. * @param {Function} callback 回调函数,必须
  4869. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  4870. * @return {Object} data 返回的数据
  4871. */
  4872. function deleteBucketCors(params, callback) {
  4873. submitRequest.call(this, {
  4874. Action: 'name/cos:DeleteBucketCORS',
  4875. method: 'DELETE',
  4876. Bucket: params.Bucket,
  4877. Region: params.Region,
  4878. headers: params.Headers,
  4879. action: 'cors'
  4880. }, function (err, data) {
  4881. if (err && err.statusCode === 204) {
  4882. return callback(null, { statusCode: err.statusCode });
  4883. } else if (err) {
  4884. return callback(err);
  4885. }
  4886. callback(null, {
  4887. statusCode: data.statusCode || err.statusCode,
  4888. headers: data.headers
  4889. });
  4890. });
  4891. }
  4892. /**
  4893. * 获取 Bucket 的 地域信息
  4894. * @param {Object} params 参数对象,必须
  4895. * @param {String} params.Bucket Bucket名称,必须
  4896. * @param {String} params.Region 地域名称,必须
  4897. * @param {Function} callback 回调函数,必须
  4898. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  4899. * @return {Object} data 返回数据,包含地域信息 LocationConstraint
  4900. */
  4901. function getBucketLocation(params, callback) {
  4902. submitRequest.call(this, {
  4903. Action: 'name/cos:GetBucketLocation',
  4904. method: 'GET',
  4905. Bucket: params.Bucket,
  4906. Region: params.Region,
  4907. headers: params.Headers,
  4908. action: 'location'
  4909. }, callback);
  4910. }
  4911. function putBucketPolicy(params, callback) {
  4912. var Policy = params['Policy'];
  4913. try {
  4914. if (typeof Policy === 'string') Policy = JSON.parse(Policy);
  4915. } catch (e) {}
  4916. if (!Policy || typeof Policy === 'string') return callback(util.error(new Error('Policy format error')));
  4917. var PolicyStr = JSON.stringify(Policy);
  4918. if (!Policy.version) Policy.version = '2.0';
  4919. var headers = params.Headers;
  4920. headers['Content-Type'] = 'application/json';
  4921. headers['Content-MD5'] = util.binaryBase64(util.md5(PolicyStr));
  4922. submitRequest.call(this, {
  4923. Action: 'name/cos:PutBucketPolicy',
  4924. method: 'PUT',
  4925. Bucket: params.Bucket,
  4926. Region: params.Region,
  4927. action: 'policy',
  4928. body: PolicyStr,
  4929. headers: headers
  4930. }, function (err, data) {
  4931. if (err && err.statusCode === 204) {
  4932. return callback(null, { statusCode: err.statusCode });
  4933. } else if (err) {
  4934. return callback(err);
  4935. }
  4936. callback(null, {
  4937. statusCode: data.statusCode,
  4938. headers: data.headers
  4939. });
  4940. });
  4941. }
  4942. /**
  4943. * 获取 Bucket 的读取权限策略
  4944. * @param {Object} params 参数对象,必须
  4945. * @param {String} params.Bucket Bucket名称,必须
  4946. * @param {String} params.Region 地域名称,必须
  4947. * @param {Function} callback 回调函数,必须
  4948. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  4949. * @return {Object} data 返回数据
  4950. */
  4951. function getBucketPolicy(params, callback) {
  4952. submitRequest.call(this, {
  4953. Action: 'name/cos:GetBucketPolicy',
  4954. method: 'GET',
  4955. Bucket: params.Bucket,
  4956. Region: params.Region,
  4957. headers: params.Headers,
  4958. action: 'policy',
  4959. rawBody: true
  4960. }, function (err, data) {
  4961. if (err) {
  4962. if (err.statusCode && err.statusCode === 403) {
  4963. return callback(util.error(err, { ErrorStatus: 'Access Denied' }));
  4964. }
  4965. if (err.statusCode && err.statusCode === 405) {
  4966. return callback(util.error(err, { ErrorStatus: 'Method Not Allowed' }));
  4967. }
  4968. if (err.statusCode && err.statusCode === 404) {
  4969. return callback(util.error(err, { ErrorStatus: 'Policy Not Found' }));
  4970. }
  4971. return callback(err);
  4972. }
  4973. var Policy = {};
  4974. try {
  4975. Policy = JSON.parse(data.body);
  4976. } catch (e) {}
  4977. callback(null, {
  4978. Policy: Policy,
  4979. statusCode: data.statusCode,
  4980. headers: data.headers
  4981. });
  4982. });
  4983. }
  4984. /**
  4985. * 删除 Bucket 的 跨域设置
  4986. * @param {Object} params 参数对象,必须
  4987. * @param {String} params.Bucket Bucket名称,必须
  4988. * @param {String} params.Region 地域名称,必须
  4989. * @param {Function} callback 回调函数,必须
  4990. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  4991. * @return {Object} data 返回的数据
  4992. */
  4993. function deleteBucketPolicy(params, callback) {
  4994. submitRequest.call(this, {
  4995. Action: 'name/cos:DeleteBucketPolicy',
  4996. method: 'DELETE',
  4997. Bucket: params.Bucket,
  4998. Region: params.Region,
  4999. headers: params.Headers,
  5000. action: 'policy'
  5001. }, function (err, data) {
  5002. if (err && err.statusCode === 204) {
  5003. return callback(null, { statusCode: err.statusCode });
  5004. } else if (err) {
  5005. return callback(err);
  5006. }
  5007. callback(null, {
  5008. statusCode: data.statusCode || err.statusCode,
  5009. headers: data.headers
  5010. });
  5011. });
  5012. }
  5013. /**
  5014. * 设置 Bucket 的标签
  5015. * @param {Object} params 参数对象,必须
  5016. * @param {String} params.Bucket Bucket名称,必须
  5017. * @param {String} params.Region 地域名称,必须
  5018. * @param {Array} params.TagSet 标签设置,必须
  5019. * @param {Function} callback 回调函数,必须
  5020. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5021. * @return {Object} data 返回数据
  5022. */
  5023. function putBucketTagging(params, callback) {
  5024. var Tagging = params['Tagging'] || {};
  5025. var Tags = Tagging.TagSet || Tagging.Tags || params['Tags'] || [];
  5026. Tags = util.clone(util.isArray(Tags) ? Tags : [Tags]);
  5027. var xml = util.json2xml({ Tagging: { TagSet: { Tag: Tags } } });
  5028. var headers = params.Headers;
  5029. headers['Content-Type'] = 'application/xml';
  5030. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  5031. submitRequest.call(this, {
  5032. Action: 'name/cos:PutBucketTagging',
  5033. method: 'PUT',
  5034. Bucket: params.Bucket,
  5035. Region: params.Region,
  5036. body: xml,
  5037. action: 'tagging',
  5038. headers: headers
  5039. }, function (err, data) {
  5040. if (err && err.statusCode === 204) {
  5041. return callback(null, { statusCode: err.statusCode });
  5042. } else if (err) {
  5043. return callback(err);
  5044. }
  5045. callback(null, {
  5046. statusCode: data.statusCode,
  5047. headers: data.headers
  5048. });
  5049. });
  5050. }
  5051. /**
  5052. * 获取 Bucket 的标签设置
  5053. * @param {Object} params 参数对象,必须
  5054. * @param {String} params.Bucket Bucket名称,必须
  5055. * @param {String} params.Region 地域名称,必须
  5056. * @param {Function} callback 回调函数,必须
  5057. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5058. * @return {Object} data 返回数据
  5059. */
  5060. function getBucketTagging(params, callback) {
  5061. submitRequest.call(this, {
  5062. Action: 'name/cos:GetBucketTagging',
  5063. method: 'GET',
  5064. Bucket: params.Bucket,
  5065. Region: params.Region,
  5066. headers: params.Headers,
  5067. action: 'tagging'
  5068. }, function (err, data) {
  5069. if (err) {
  5070. if (err.statusCode === 404 && err.error && (err.error === "Not Found" || err.error.Code === 'NoSuchTagSet')) {
  5071. var result = {
  5072. Tags: [],
  5073. statusCode: err.statusCode
  5074. };
  5075. err.headers && (result.headers = err.headers);
  5076. callback(null, result);
  5077. } else {
  5078. callback(err);
  5079. }
  5080. return;
  5081. }
  5082. var Tags = [];
  5083. try {
  5084. Tags = data.Tagging.TagSet.Tag || [];
  5085. } catch (e) {}
  5086. Tags = util.clone(util.isArray(Tags) ? Tags : [Tags]);
  5087. callback(null, {
  5088. Tags: Tags,
  5089. statusCode: data.statusCode,
  5090. headers: data.headers
  5091. });
  5092. });
  5093. }
  5094. /**
  5095. * 删除 Bucket 的 标签设置
  5096. * @param {Object} params 参数对象,必须
  5097. * @param {String} params.Bucket Bucket名称,必须
  5098. * @param {String} params.Region 地域名称,必须
  5099. * @param {Function} callback 回调函数,必须
  5100. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5101. * @return {Object} data 返回的数据
  5102. */
  5103. function deleteBucketTagging(params, callback) {
  5104. submitRequest.call(this, {
  5105. Action: 'name/cos:DeleteBucketTagging',
  5106. method: 'DELETE',
  5107. Bucket: params.Bucket,
  5108. Region: params.Region,
  5109. headers: params.Headers,
  5110. action: 'tagging'
  5111. }, function (err, data) {
  5112. if (err && err.statusCode === 204) {
  5113. return callback(null, { statusCode: err.statusCode });
  5114. } else if (err) {
  5115. return callback(err);
  5116. }
  5117. callback(null, {
  5118. statusCode: data.statusCode,
  5119. headers: data.headers
  5120. });
  5121. });
  5122. }
  5123. function putBucketLifecycle(params, callback) {
  5124. var LifecycleConfiguration = params['LifecycleConfiguration'] || {};
  5125. var Rules = LifecycleConfiguration.Rules || params.Rules || [];
  5126. Rules = util.clone(Rules);
  5127. var xml = util.json2xml({ LifecycleConfiguration: { Rule: Rules } });
  5128. var headers = params.Headers;
  5129. headers['Content-Type'] = 'application/xml';
  5130. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  5131. submitRequest.call(this, {
  5132. Action: 'name/cos:PutBucketLifecycle',
  5133. method: 'PUT',
  5134. Bucket: params.Bucket,
  5135. Region: params.Region,
  5136. body: xml,
  5137. action: 'lifecycle',
  5138. headers: headers
  5139. }, function (err, data) {
  5140. if (err && err.statusCode === 204) {
  5141. return callback(null, { statusCode: err.statusCode });
  5142. } else if (err) {
  5143. return callback(err);
  5144. }
  5145. callback(null, {
  5146. statusCode: data.statusCode,
  5147. headers: data.headers
  5148. });
  5149. });
  5150. }
  5151. function getBucketLifecycle(params, callback) {
  5152. submitRequest.call(this, {
  5153. Action: 'name/cos:GetBucketLifecycle',
  5154. method: 'GET',
  5155. Bucket: params.Bucket,
  5156. Region: params.Region,
  5157. headers: params.Headers,
  5158. action: 'lifecycle'
  5159. }, function (err, data) {
  5160. if (err) {
  5161. if (err.statusCode === 404 && err.error && err.error.Code === 'NoSuchLifecycleConfiguration') {
  5162. var result = {
  5163. Rules: [],
  5164. statusCode: err.statusCode
  5165. };
  5166. err.headers && (result.headers = err.headers);
  5167. callback(null, result);
  5168. } else {
  5169. callback(err);
  5170. }
  5171. return;
  5172. }
  5173. var Rules = [];
  5174. try {
  5175. Rules = data.LifecycleConfiguration.Rule || [];
  5176. } catch (e) {}
  5177. Rules = util.clone(util.isArray(Rules) ? Rules : [Rules]);
  5178. callback(null, {
  5179. Rules: Rules,
  5180. statusCode: data.statusCode,
  5181. headers: data.headers
  5182. });
  5183. });
  5184. }
  5185. function deleteBucketLifecycle(params, callback) {
  5186. submitRequest.call(this, {
  5187. Action: 'name/cos:DeleteBucketLifecycle',
  5188. method: 'DELETE',
  5189. Bucket: params.Bucket,
  5190. Region: params.Region,
  5191. headers: params.Headers,
  5192. action: 'lifecycle'
  5193. }, function (err, data) {
  5194. if (err && err.statusCode === 204) {
  5195. return callback(null, { statusCode: err.statusCode });
  5196. } else if (err) {
  5197. return callback(err);
  5198. }
  5199. callback(null, {
  5200. statusCode: data.statusCode,
  5201. headers: data.headers
  5202. });
  5203. });
  5204. }
  5205. function putBucketVersioning(params, callback) {
  5206. if (!params['VersioningConfiguration']) {
  5207. callback(util.error(new Error('missing param VersioningConfiguration')));
  5208. return;
  5209. }
  5210. var VersioningConfiguration = params['VersioningConfiguration'] || {};
  5211. var xml = util.json2xml({ VersioningConfiguration: VersioningConfiguration });
  5212. var headers = params.Headers;
  5213. headers['Content-Type'] = 'application/xml';
  5214. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  5215. submitRequest.call(this, {
  5216. Action: 'name/cos:PutBucketVersioning',
  5217. method: 'PUT',
  5218. Bucket: params.Bucket,
  5219. Region: params.Region,
  5220. body: xml,
  5221. action: 'versioning',
  5222. headers: headers
  5223. }, function (err, data) {
  5224. if (err && err.statusCode === 204) {
  5225. return callback(null, { statusCode: err.statusCode });
  5226. } else if (err) {
  5227. return callback(err);
  5228. }
  5229. callback(null, {
  5230. statusCode: data.statusCode,
  5231. headers: data.headers
  5232. });
  5233. });
  5234. }
  5235. function getBucketVersioning(params, callback) {
  5236. submitRequest.call(this, {
  5237. Action: 'name/cos:GetBucketVersioning',
  5238. method: 'GET',
  5239. Bucket: params.Bucket,
  5240. Region: params.Region,
  5241. headers: params.Headers,
  5242. action: 'versioning'
  5243. }, function (err, data) {
  5244. if (!err) {
  5245. !data.VersioningConfiguration && (data.VersioningConfiguration = {});
  5246. }
  5247. callback(err, data);
  5248. });
  5249. }
  5250. function putBucketReplication(params, callback) {
  5251. var ReplicationConfiguration = util.clone(params.ReplicationConfiguration);
  5252. var xml = util.json2xml({ ReplicationConfiguration: ReplicationConfiguration });
  5253. xml = xml.replace(/<(\/?)Rules>/ig, '<$1Rule>');
  5254. xml = xml.replace(/<(\/?)Tags>/ig, '<$1Tag>');
  5255. var headers = params.Headers;
  5256. headers['Content-Type'] = 'application/xml';
  5257. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  5258. submitRequest.call(this, {
  5259. Action: 'name/cos:PutBucketReplication',
  5260. method: 'PUT',
  5261. Bucket: params.Bucket,
  5262. Region: params.Region,
  5263. body: xml,
  5264. action: 'replication',
  5265. headers: headers
  5266. }, function (err, data) {
  5267. if (err && err.statusCode === 204) {
  5268. return callback(null, { statusCode: err.statusCode });
  5269. } else if (err) {
  5270. return callback(err);
  5271. }
  5272. callback(null, {
  5273. statusCode: data.statusCode,
  5274. headers: data.headers
  5275. });
  5276. });
  5277. }
  5278. function getBucketReplication(params, callback) {
  5279. submitRequest.call(this, {
  5280. Action: 'name/cos:GetBucketReplication',
  5281. method: 'GET',
  5282. Bucket: params.Bucket,
  5283. Region: params.Region,
  5284. headers: params.Headers,
  5285. action: 'replication'
  5286. }, function (err, data) {
  5287. if (err) {
  5288. if (err.statusCode === 404 && err.error && (err.error === 'Not Found' || err.error.Code === 'ReplicationConfigurationnotFoundError')) {
  5289. var result = {
  5290. ReplicationConfiguration: { Rules: [] },
  5291. statusCode: err.statusCode
  5292. };
  5293. err.headers && (result.headers = err.headers);
  5294. callback(null, result);
  5295. } else {
  5296. callback(err);
  5297. }
  5298. return;
  5299. }
  5300. !data.ReplicationConfiguration && (data.ReplicationConfiguration = {});
  5301. if (data.ReplicationConfiguration.Rule) {
  5302. data.ReplicationConfiguration.Rules = util.makeArray(data.ReplicationConfiguration.Rule);
  5303. delete data.ReplicationConfiguration.Rule;
  5304. }
  5305. callback(err, data);
  5306. });
  5307. }
  5308. function deleteBucketReplication(params, callback) {
  5309. submitRequest.call(this, {
  5310. Action: 'name/cos:DeleteBucketReplication',
  5311. method: 'DELETE',
  5312. Bucket: params.Bucket,
  5313. Region: params.Region,
  5314. headers: params.Headers,
  5315. action: 'replication'
  5316. }, function (err, data) {
  5317. if (err && err.statusCode === 204) {
  5318. return callback(null, { statusCode: err.statusCode });
  5319. } else if (err) {
  5320. return callback(err);
  5321. }
  5322. callback(null, {
  5323. statusCode: data.statusCode,
  5324. headers: data.headers
  5325. });
  5326. });
  5327. }
  5328. /**
  5329. * 设置 Bucket 静态网站配置信息
  5330. * @param {Object} params 参数对象,必须
  5331. * @param {String} params.Bucket Bucket名称,必须
  5332. * @param {String} params.Region 地域名称,必须
  5333. * @param {Object} params.WebsiteConfiguration 地域名称,必须
  5334. * @param {Object} WebsiteConfiguration.IndexDocument 索引文档,必须
  5335. * @param {Object} WebsiteConfiguration.ErrorDocument 错误文档,非必须
  5336. * @param {Object} WebsiteConfiguration.RedirectAllRequestsTo 重定向所有请求,非必须
  5337. * @param {Array} params.RoutingRules 重定向规则,非必须
  5338. * @param {Function} callback 回调函数,必须
  5339. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5340. * @return {Object} data 返回数据
  5341. */
  5342. function putBucketWebsite(params, callback) {
  5343. if (!params['WebsiteConfiguration']) {
  5344. callback(util.error(new Error('missing param WebsiteConfiguration')));
  5345. return;
  5346. }
  5347. var WebsiteConfiguration = util.clone(params['WebsiteConfiguration'] || {});
  5348. var RoutingRules = WebsiteConfiguration['RoutingRules'] || WebsiteConfiguration['RoutingRule'] || [];
  5349. RoutingRules = util.isArray(RoutingRules) ? RoutingRules : [RoutingRules];
  5350. delete WebsiteConfiguration.RoutingRule;
  5351. delete WebsiteConfiguration.RoutingRules;
  5352. if (RoutingRules.length) WebsiteConfiguration.RoutingRules = { RoutingRule: RoutingRules };
  5353. var xml = util.json2xml({ WebsiteConfiguration: WebsiteConfiguration });
  5354. var headers = params.Headers;
  5355. headers['Content-Type'] = 'application/xml';
  5356. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  5357. submitRequest.call(this, {
  5358. Action: 'name/cos:PutBucketWebsite',
  5359. method: 'PUT',
  5360. Bucket: params.Bucket,
  5361. Region: params.Region,
  5362. body: xml,
  5363. action: 'website',
  5364. headers: headers
  5365. }, function (err, data) {
  5366. if (err && err.statusCode === 204) {
  5367. return callback(null, { statusCode: err.statusCode });
  5368. } else if (err) {
  5369. return callback(err);
  5370. }
  5371. callback(null, {
  5372. statusCode: data.statusCode,
  5373. headers: data.headers
  5374. });
  5375. });
  5376. }
  5377. /**
  5378. * 获取 Bucket 的静态网站配置信息
  5379. * @param {Object} params 参数对象,必须
  5380. * @param {String} params.Bucket Bucket名称,必须
  5381. * @param {String} params.Region 地域名称,必须
  5382. * @param {Function} callback 回调函数,必须
  5383. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5384. * @return {Object} data 返回数据
  5385. */
  5386. function getBucketWebsite(params, callback) {
  5387. submitRequest.call(this, {
  5388. Action: 'name/cos:GetBucketWebsite',
  5389. method: 'GET',
  5390. Bucket: params.Bucket,
  5391. Region: params.Region,
  5392. Key: params.Key,
  5393. headers: params.Headers,
  5394. action: 'website'
  5395. }, function (err, data) {
  5396. if (err) {
  5397. if (err.statusCode === 404 && err.error.Code === 'NoSuchWebsiteConfiguration') {
  5398. var result = {
  5399. WebsiteConfiguration: {},
  5400. statusCode: err.statusCode
  5401. };
  5402. err.headers && (result.headers = err.headers);
  5403. callback(null, result);
  5404. } else {
  5405. callback(err);
  5406. }
  5407. return;
  5408. }
  5409. var WebsiteConfiguration = data.WebsiteConfiguration || {};
  5410. if (WebsiteConfiguration['RoutingRules']) {
  5411. var RoutingRules = util.clone(WebsiteConfiguration['RoutingRules'].RoutingRule || []);
  5412. RoutingRules = util.makeArray(RoutingRules);
  5413. WebsiteConfiguration.RoutingRules = RoutingRules;
  5414. }
  5415. callback(null, {
  5416. WebsiteConfiguration: WebsiteConfiguration,
  5417. statusCode: data.statusCode,
  5418. headers: data.headers
  5419. });
  5420. });
  5421. }
  5422. /**
  5423. * 删除 Bucket 的静态网站配置
  5424. * @param {Object} params 参数对象,必须
  5425. * @param {String} params.Bucket Bucket名称,必须
  5426. * @param {String} params.Region 地域名称,必须
  5427. * @param {Function} callback 回调函数,必须
  5428. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5429. * @return {Object} data 返回数据
  5430. */
  5431. function deleteBucketWebsite(params, callback) {
  5432. submitRequest.call(this, {
  5433. Action: 'name/cos:DeleteBucketWebsite',
  5434. method: 'DELETE',
  5435. Bucket: params.Bucket,
  5436. Region: params.Region,
  5437. headers: params.Headers,
  5438. action: 'website'
  5439. }, function (err, data) {
  5440. if (err && err.statusCode === 204) {
  5441. return callback(null, { statusCode: err.statusCode });
  5442. } else if (err) {
  5443. return callback(err);
  5444. }
  5445. callback(null, {
  5446. statusCode: data.statusCode,
  5447. headers: data.headers
  5448. });
  5449. });
  5450. }
  5451. /**
  5452. * 设置 Bucket 的防盗链白名单或者黑名单
  5453. * @param {Object} params 参数对象,必须
  5454. * @param {String} params.Bucket Bucket名称,必须
  5455. * @param {String} params.Region 地域名称,必须
  5456. * @param {Object} params.RefererConfiguration 地域名称,必须
  5457. * @param {String} RefererConfiguration.Status 是否开启防盗链,枚举值:Enabled、Disabled
  5458. * @param {String} RefererConfiguration.RefererType 防盗链类型,枚举值:Black-List、White-List,必须
  5459. * @param {Array} RefererConfiguration.DomianList.Domain 生效域名,必须
  5460. * @param {String} RefererConfiguration.EmptyReferConfiguration ,非必须
  5461. * @param {Function} callback 回调函数,必须
  5462. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5463. * @return {Object} data 返回数据
  5464. */
  5465. function putBucketReferer(params, callback) {
  5466. if (!params['RefererConfiguration']) {
  5467. callback(util.error(new Error('missing param RefererConfiguration')));
  5468. return;
  5469. }
  5470. var RefererConfiguration = util.clone(params['RefererConfiguration'] || {});
  5471. var DomainList = RefererConfiguration['DomainList'] || {};
  5472. var Domains = DomainList['Domains'] || DomainList['Domain'] || [];
  5473. Domains = util.isArray(Domains) ? Domains : [Domains];
  5474. if (Domains.length) RefererConfiguration.DomainList = { Domain: Domains };
  5475. var xml = util.json2xml({ RefererConfiguration: RefererConfiguration });
  5476. var headers = params.Headers;
  5477. headers['Content-Type'] = 'application/xml';
  5478. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  5479. submitRequest.call(this, {
  5480. Action: 'name/cos:PutBucketReferer',
  5481. method: 'PUT',
  5482. Bucket: params.Bucket,
  5483. Region: params.Region,
  5484. body: xml,
  5485. action: 'referer',
  5486. headers: headers
  5487. }, function (err, data) {
  5488. if (err && err.statusCode === 204) {
  5489. return callback(null, { statusCode: err.statusCode });
  5490. } else if (err) {
  5491. return callback(err);
  5492. }
  5493. callback(null, {
  5494. statusCode: data.statusCode,
  5495. headers: data.headers
  5496. });
  5497. });
  5498. }
  5499. /**
  5500. * 获取 Bucket 的防盗链白名单或者黑名单
  5501. * @param {Object} params 参数对象,必须
  5502. * @param {String} params.Bucket Bucket名称,必须
  5503. * @param {String} params.Region 地域名称,必须
  5504. * @param {Function} callback 回调函数,必须
  5505. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5506. * @return {Object} data 返回数据
  5507. */
  5508. function getBucketReferer(params, callback) {
  5509. submitRequest.call(this, {
  5510. Action: 'name/cos:GetBucketReferer',
  5511. method: 'GET',
  5512. Bucket: params.Bucket,
  5513. Region: params.Region,
  5514. Key: params.Key,
  5515. headers: params.Headers,
  5516. action: 'referer'
  5517. }, function (err, data) {
  5518. if (err) {
  5519. if (err.statusCode === 404 && err.error.Code === 'NoSuchRefererConfiguration') {
  5520. var result = {
  5521. WebsiteConfiguration: {},
  5522. statusCode: err.statusCode
  5523. };
  5524. err.headers && (result.headers = err.headers);
  5525. callback(null, result);
  5526. } else {
  5527. callback(err);
  5528. }
  5529. return;
  5530. }
  5531. var RefererConfiguration = data.RefererConfiguration || {};
  5532. if (RefererConfiguration['DomainList']) {
  5533. var Domains = util.clone(RefererConfiguration['DomainList'].Domain || []);
  5534. Domains = util.makeArray(Domains);
  5535. RefererConfiguration.DomainList = { Domains: Domains };
  5536. }
  5537. callback(null, {
  5538. RefererConfiguration: RefererConfiguration,
  5539. statusCode: data.statusCode,
  5540. headers: data.headers
  5541. });
  5542. });
  5543. }
  5544. /**
  5545. * 设置 Bucket 自定义域名
  5546. * @param {Object} params 参数对象,必须
  5547. * @param {String} params.Bucket Bucket名称,必须
  5548. * @param {String} params.Region 地域名称,必须
  5549. * @param {Function} callback 回调函数,必须
  5550. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5551. * @return {Object} data 返回数据
  5552. */
  5553. function putBucketDomain(params, callback) {
  5554. var DomainConfiguration = params['DomainConfiguration'] || {};
  5555. var DomainRule = DomainConfiguration.DomainRule || params.DomainRule || [];
  5556. DomainRule = util.clone(DomainRule);
  5557. var xml = util.json2xml({ DomainConfiguration: { DomainRule: DomainRule } });
  5558. var headers = params.Headers;
  5559. headers['Content-Type'] = 'application/xml';
  5560. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  5561. submitRequest.call(this, {
  5562. Action: 'name/cos:PutBucketDomain',
  5563. method: 'PUT',
  5564. Bucket: params.Bucket,
  5565. Region: params.Region,
  5566. body: xml,
  5567. action: 'domain',
  5568. headers: headers
  5569. }, function (err, data) {
  5570. if (err && err.statusCode === 204) {
  5571. return callback(null, { statusCode: err.statusCode });
  5572. } else if (err) {
  5573. return callback(err);
  5574. }
  5575. callback(null, {
  5576. statusCode: data.statusCode,
  5577. headers: data.headers
  5578. });
  5579. });
  5580. }
  5581. /**
  5582. * 获取 Bucket 的自定义域名
  5583. * @param {Object} params 参数对象,必须
  5584. * @param {String} params.Bucket Bucket名称,必须
  5585. * @param {String} params.Region 地域名称,必须
  5586. * @param {Function} callback 回调函数,必须
  5587. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5588. * @return {Object} data 返回数据
  5589. */
  5590. function getBucketDomain(params, callback) {
  5591. submitRequest.call(this, {
  5592. Action: 'name/cos:GetBucketDomain',
  5593. method: 'GET',
  5594. Bucket: params.Bucket,
  5595. Region: params.Region,
  5596. headers: params.Headers,
  5597. action: 'domain'
  5598. }, function (err, data) {
  5599. if (err) return callback(err);
  5600. var DomainRule = [];
  5601. try {
  5602. DomainRule = data.DomainConfiguration.DomainRule || [];
  5603. } catch (e) {}
  5604. DomainRule = util.clone(util.isArray(DomainRule) ? DomainRule : [DomainRule]);
  5605. callback(null, {
  5606. DomainRule: DomainRule,
  5607. statusCode: data.statusCode,
  5608. headers: data.headers
  5609. });
  5610. });
  5611. }
  5612. /**
  5613. * 删除 Bucket 自定义域名
  5614. * @param {Object} params 参数对象,必须
  5615. * @param {String} params.Bucket Bucket名称,必须
  5616. * @param {String} params.Region 地域名称,必须
  5617. * @param {Function} callback 回调函数,必须
  5618. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5619. * @return {Object} data 返回数据
  5620. */
  5621. function deleteBucketDomain(params, callback) {
  5622. submitRequest.call(this, {
  5623. Action: 'name/cos:DeleteBucketDomain',
  5624. method: 'DELETE',
  5625. Bucket: params.Bucket,
  5626. Region: params.Region,
  5627. headers: params.Headers,
  5628. action: 'domain'
  5629. }, function (err, data) {
  5630. if (err && err.statusCode === 204) {
  5631. return callback(null, { statusCode: err.statusCode });
  5632. } else if (err) {
  5633. return callback(err);
  5634. }
  5635. callback(null, {
  5636. statusCode: data.statusCode,
  5637. headers: data.headers
  5638. });
  5639. });
  5640. }
  5641. /**
  5642. * 设置 Bucket 的回源
  5643. * @param {Object} params 参数对象,必须
  5644. * @param {String} params.Bucket Bucket名称,必须
  5645. * @param {String} params.Region 地域名称,必须
  5646. * @param {Function} callback 回调函数,必须
  5647. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5648. * @return {Object} data 返回数据
  5649. */
  5650. function putBucketOrigin(params, callback) {
  5651. var OriginConfiguration = params['OriginConfiguration'] || {};
  5652. var OriginRule = OriginConfiguration.OriginRule || params.OriginRule || [];
  5653. OriginRule = util.clone(OriginRule);
  5654. var xml = util.json2xml({ OriginConfiguration: { OriginRule: OriginRule } });
  5655. var headers = params.Headers;
  5656. headers['Content-Type'] = 'application/xml';
  5657. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  5658. submitRequest.call(this, {
  5659. Action: 'name/cos:PutBucketOrigin',
  5660. method: 'PUT',
  5661. Bucket: params.Bucket,
  5662. Region: params.Region,
  5663. body: xml,
  5664. action: 'origin',
  5665. headers: headers
  5666. }, function (err, data) {
  5667. if (err && err.statusCode === 204) {
  5668. return callback(null, { statusCode: err.statusCode });
  5669. } else if (err) {
  5670. return callback(err);
  5671. }
  5672. callback(null, {
  5673. statusCode: data.statusCode,
  5674. headers: data.headers
  5675. });
  5676. });
  5677. }
  5678. /**
  5679. * 获取 Bucket 的回源
  5680. * @param {Object} params 参数对象,必须
  5681. * @param {String} params.Bucket Bucket名称,必须
  5682. * @param {String} params.Region 地域名称,必须
  5683. * @param {Function} callback 回调函数,必须
  5684. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5685. * @return {Object} data 返回数据
  5686. */
  5687. function getBucketOrigin(params, callback) {
  5688. submitRequest.call(this, {
  5689. Action: 'name/cos:GetBucketOrigin',
  5690. method: 'GET',
  5691. Bucket: params.Bucket,
  5692. Region: params.Region,
  5693. headers: params.Headers,
  5694. action: 'origin'
  5695. }, function (err, data) {
  5696. if (err) return callback(err);
  5697. var OriginRule = [];
  5698. try {
  5699. OriginRule = data.OriginConfiguration.OriginRule || [];
  5700. } catch (e) {}
  5701. OriginRule = util.clone(util.isArray(OriginRule) ? OriginRule : [OriginRule]);
  5702. callback(null, {
  5703. OriginRule: OriginRule,
  5704. statusCode: data.statusCode,
  5705. headers: data.headers
  5706. });
  5707. });
  5708. }
  5709. /**
  5710. * 删除 Bucket 的回源
  5711. * @param {Object} params 参数对象,必须
  5712. * @param {String} params.Bucket Bucket名称,必须
  5713. * @param {String} params.Region 地域名称,必须
  5714. * @param {Function} callback 回调函数,必须
  5715. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5716. * @return {Object} data 返回数据
  5717. */
  5718. function deleteBucketOrigin(params, callback) {
  5719. submitRequest.call(this, {
  5720. Action: 'name/cos:DeleteBucketOrigin',
  5721. method: 'DELETE',
  5722. Bucket: params.Bucket,
  5723. Region: params.Region,
  5724. headers: params.Headers,
  5725. action: 'origin'
  5726. }, function (err, data) {
  5727. if (err && err.statusCode === 204) {
  5728. return callback(null, { statusCode: err.statusCode });
  5729. } else if (err) {
  5730. return callback(err);
  5731. }
  5732. callback(null, {
  5733. statusCode: data.statusCode,
  5734. headers: data.headers
  5735. });
  5736. });
  5737. }
  5738. /**
  5739. * 设置 Bucket 的日志记录
  5740. * @param {Object} params 参数对象,必须
  5741. * @param {String} params.Bucket Bucket名称,必须
  5742. * @param {String} params.Region 地域名称,必须
  5743. * @param {(Object|String)} params.BucketLoggingStatus 说明日志记录配置的状态,如果无子节点信息则意为关闭日志记录,必须
  5744. * @param {Function} callback 回调函数,必须
  5745. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5746. * @return {Object} data 返回数据
  5747. */
  5748. function putBucketLogging(params, callback) {
  5749. var xml = util.json2xml({
  5750. BucketLoggingStatus: params['BucketLoggingStatus'] || ''
  5751. });
  5752. var headers = params.Headers;
  5753. headers['Content-Type'] = 'application/xml';
  5754. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  5755. submitRequest.call(this, {
  5756. Action: 'name/cos:PutBucketLogging',
  5757. method: 'PUT',
  5758. Bucket: params.Bucket,
  5759. Region: params.Region,
  5760. body: xml,
  5761. action: 'logging',
  5762. headers: headers
  5763. }, function (err, data) {
  5764. if (err && err.statusCode === 204) {
  5765. return callback(null, { statusCode: err.statusCode });
  5766. } else if (err) {
  5767. return callback(err);
  5768. }
  5769. callback(null, {
  5770. statusCode: data.statusCode,
  5771. headers: data.headers
  5772. });
  5773. });
  5774. }
  5775. /**
  5776. * 获取 Bucket 的日志记录
  5777. * @param {Object} params 参数对象,必须
  5778. * @param {String} params.Bucket Bucket名称,必须
  5779. * @param {String} params.Region 地域名称,必须
  5780. * @param {Function} callback 回调函数,必须
  5781. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5782. * @return {Object} data 返回数据
  5783. */
  5784. function getBucketLogging(params, callback) {
  5785. submitRequest.call(this, {
  5786. Action: 'name/cos:GetBucketLogging',
  5787. method: 'GET',
  5788. Bucket: params.Bucket,
  5789. Region: params.Region,
  5790. headers: params.Headers,
  5791. action: 'logging'
  5792. }, function (err, data) {
  5793. if (err) return callback(err);
  5794. callback(null, {
  5795. BucketLoggingStatus: data.BucketLoggingStatus,
  5796. statusCode: data.statusCode,
  5797. headers: data.headers
  5798. });
  5799. });
  5800. }
  5801. /**
  5802. * 创建/编辑 Bucket 的清单任务
  5803. * @param {Object} params 参数对象,必须
  5804. * @param {String} params.Bucket Bucket名称,必须
  5805. * @param {String} params.Region 地域名称,必须
  5806. * @param {String} params.Id 清单任务的名称,必须
  5807. * @param {Object} params.InventoryConfiguration 包含清单的配置参数,必须
  5808. * @param {Function} callback 回调函数,必须
  5809. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5810. * @return {Object} data 返回数据
  5811. */
  5812. function putBucketInventory(params, callback) {
  5813. var InventoryConfiguration = util.clone(params['InventoryConfiguration']);
  5814. if (InventoryConfiguration.OptionalFields) {
  5815. var Field = InventoryConfiguration.OptionalFields || [];
  5816. InventoryConfiguration.OptionalFields = {
  5817. Field: Field
  5818. };
  5819. }
  5820. if (InventoryConfiguration.Destination && InventoryConfiguration.Destination.COSBucketDestination && InventoryConfiguration.Destination.COSBucketDestination.Encryption) {
  5821. var Encryption = InventoryConfiguration.Destination.COSBucketDestination.Encryption;
  5822. if (Object.keys(Encryption).indexOf('SSECOS') > -1) {
  5823. Encryption['SSE-COS'] = Encryption['SSECOS'];
  5824. delete Encryption['SSECOS'];
  5825. }
  5826. }
  5827. var xml = util.json2xml({
  5828. InventoryConfiguration: InventoryConfiguration
  5829. });
  5830. var headers = params.Headers;
  5831. headers['Content-Type'] = 'application/xml';
  5832. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  5833. submitRequest.call(this, {
  5834. Action: 'name/cos:PutBucketInventory',
  5835. method: 'PUT',
  5836. Bucket: params.Bucket,
  5837. Region: params.Region,
  5838. body: xml,
  5839. action: 'inventory',
  5840. qs: {
  5841. id: params['Id']
  5842. },
  5843. headers: headers
  5844. }, function (err, data) {
  5845. if (err && err.statusCode === 204) {
  5846. return callback(null, { statusCode: err.statusCode });
  5847. } else if (err) {
  5848. return callback(err);
  5849. }
  5850. callback(null, {
  5851. statusCode: data.statusCode,
  5852. headers: data.headers
  5853. });
  5854. });
  5855. }
  5856. /**
  5857. * 获取 Bucket 的清单任务信息
  5858. * @param {Object} params 参数对象,必须
  5859. * @param {String} params.Bucket Bucket名称,必须
  5860. * @param {String} params.Region 地域名称,必须
  5861. * @param {String} params.Id 清单任务的名称,必须
  5862. * @param {Function} callback 回调函数,必须
  5863. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5864. * @return {Object} data 返回数据
  5865. */
  5866. function getBucketInventory(params, callback) {
  5867. submitRequest.call(this, {
  5868. Action: 'name/cos:GetBucketInventory',
  5869. method: 'GET',
  5870. Bucket: params.Bucket,
  5871. Region: params.Region,
  5872. headers: params.Headers,
  5873. action: 'inventory',
  5874. qs: {
  5875. id: params['Id']
  5876. }
  5877. }, function (err, data) {
  5878. if (err) return callback(err);
  5879. var InventoryConfiguration = data['InventoryConfiguration'];
  5880. if (InventoryConfiguration && InventoryConfiguration.OptionalFields && InventoryConfiguration.OptionalFields.Field) {
  5881. var Field = InventoryConfiguration.OptionalFields.Field;
  5882. if (!util.isArray(Field)) {
  5883. Field = [Field];
  5884. }
  5885. InventoryConfiguration.OptionalFields = Field;
  5886. }
  5887. if (InventoryConfiguration.Destination && InventoryConfiguration.Destination.COSBucketDestination && InventoryConfiguration.Destination.COSBucketDestination.Encryption) {
  5888. var Encryption = InventoryConfiguration.Destination.COSBucketDestination.Encryption;
  5889. if (Object.keys(Encryption).indexOf('SSE-COS') > -1) {
  5890. Encryption['SSECOS'] = Encryption['SSE-COS'];
  5891. delete Encryption['SSE-COS'];
  5892. }
  5893. }
  5894. callback(null, {
  5895. InventoryConfiguration: InventoryConfiguration,
  5896. statusCode: data.statusCode,
  5897. headers: data.headers
  5898. });
  5899. });
  5900. }
  5901. /**
  5902. * 获取 Bucket 的清单任务信息
  5903. * @param {Object} params 参数对象,必须
  5904. * @param {String} params.Bucket Bucket名称,必须
  5905. * @param {String} params.Region 地域名称,必须
  5906. * @param {String} params.ContinuationToken 当 COS 响应体中 IsTruncated 为 true,且 NextContinuationToken 节点中存在参数值时,您可以将这个参数作为 continuation-token 参数值,以获取下一页的清单任务信息,非必须
  5907. * @param {Function} callback 回调函数,必须
  5908. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5909. * @return {Object} data 返回数据
  5910. */
  5911. function listBucketInventory(params, callback) {
  5912. submitRequest.call(this, {
  5913. Action: 'name/cos:ListBucketInventory',
  5914. method: 'GET',
  5915. Bucket: params.Bucket,
  5916. Region: params.Region,
  5917. headers: params.Headers,
  5918. action: 'inventory',
  5919. qs: {
  5920. 'continuation-token': params['ContinuationToken']
  5921. }
  5922. }, function (err, data) {
  5923. if (err) return callback(err);
  5924. var ListInventoryConfigurationResult = data['ListInventoryConfigurationResult'];
  5925. var InventoryConfigurations = ListInventoryConfigurationResult.InventoryConfiguration || [];
  5926. InventoryConfigurations = util.isArray(InventoryConfigurations) ? InventoryConfigurations : [InventoryConfigurations];
  5927. delete ListInventoryConfigurationResult['InventoryConfiguration'];
  5928. util.each(InventoryConfigurations, function (InventoryConfiguration) {
  5929. if (InventoryConfiguration && InventoryConfiguration.OptionalFields && InventoryConfiguration.OptionalFields.Field) {
  5930. var Field = InventoryConfiguration.OptionalFields.Field;
  5931. if (!util.isArray(Field)) {
  5932. Field = [Field];
  5933. }
  5934. InventoryConfiguration.OptionalFields = Field;
  5935. }
  5936. if (InventoryConfiguration.Destination && InventoryConfiguration.Destination.COSBucketDestination && InventoryConfiguration.Destination.COSBucketDestination.Encryption) {
  5937. var Encryption = InventoryConfiguration.Destination.COSBucketDestination.Encryption;
  5938. if (Object.keys(Encryption).indexOf('SSE-COS') > -1) {
  5939. Encryption['SSECOS'] = Encryption['SSE-COS'];
  5940. delete Encryption['SSE-COS'];
  5941. }
  5942. }
  5943. });
  5944. ListInventoryConfigurationResult.InventoryConfigurations = InventoryConfigurations;
  5945. util.extend(ListInventoryConfigurationResult, {
  5946. statusCode: data.statusCode,
  5947. headers: data.headers
  5948. });
  5949. callback(null, ListInventoryConfigurationResult);
  5950. });
  5951. }
  5952. /**
  5953. * 删除 Bucket 的清单任务
  5954. * @param {Object} params 参数对象,必须
  5955. * @param {String} params.Bucket Bucket名称,必须
  5956. * @param {String} params.Region 地域名称,必须
  5957. * @param {String} params.Id 清单任务的名称,必须
  5958. * @param {Function} callback 回调函数,必须
  5959. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  5960. * @return {Object} data 返回数据
  5961. */
  5962. function deleteBucketInventory(params, callback) {
  5963. submitRequest.call(this, {
  5964. Action: 'name/cos:DeleteBucketInventory',
  5965. method: 'DELETE',
  5966. Bucket: params.Bucket,
  5967. Region: params.Region,
  5968. headers: params.Headers,
  5969. action: 'inventory',
  5970. qs: {
  5971. id: params['Id']
  5972. }
  5973. }, function (err, data) {
  5974. if (err && err.statusCode === 204) {
  5975. return callback(null, { statusCode: err.statusCode });
  5976. } else if (err) {
  5977. return callback(err);
  5978. }
  5979. callback(null, {
  5980. statusCode: data.statusCode,
  5981. headers: data.headers
  5982. });
  5983. });
  5984. }
  5985. /* 全球加速 */
  5986. function putBucketAccelerate(params, callback) {
  5987. if (!params['AccelerateConfiguration']) {
  5988. callback(util.error(new Error('missing param AccelerateConfiguration')));
  5989. return;
  5990. }
  5991. var configuration = { AccelerateConfiguration: params.AccelerateConfiguration || {} };
  5992. var xml = util.json2xml(configuration);
  5993. var headers = {};
  5994. headers['Content-Type'] = 'application/xml';
  5995. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  5996. submitRequest.call(this, {
  5997. Action: 'name/cos:PutBucketAccelerate',
  5998. method: 'PUT',
  5999. Bucket: params.Bucket,
  6000. Region: params.Region,
  6001. body: xml,
  6002. action: 'accelerate',
  6003. headers: headers
  6004. }, function (err, data) {
  6005. if (err) return callback(err);
  6006. callback(null, {
  6007. statusCode: data.statusCode,
  6008. headers: data.headers
  6009. });
  6010. });
  6011. }
  6012. function getBucketAccelerate(params, callback) {
  6013. submitRequest.call(this, {
  6014. Action: 'name/cos:GetBucketAccelerate',
  6015. method: 'GET',
  6016. Bucket: params.Bucket,
  6017. Region: params.Region,
  6018. action: 'accelerate'
  6019. }, function (err, data) {
  6020. if (!err) {
  6021. !data.AccelerateConfiguration && (data.AccelerateConfiguration = {});
  6022. }
  6023. callback(err, data);
  6024. });
  6025. }
  6026. function putBucketEncryption(params, callback) {
  6027. var conf = params.ServerSideEncryptionConfiguration || {};
  6028. var Rules = conf.Rule || conf.Rules || [];
  6029. var xml = util.json2xml({ ServerSideEncryptionConfiguration: { Rule: Rules } });
  6030. var headers = params.Headers;
  6031. headers['Content-Type'] = 'application/xml';
  6032. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  6033. submitRequest.call(this, {
  6034. Action: 'name/cos:PutBucketEncryption',
  6035. method: 'PUT',
  6036. Bucket: params.Bucket,
  6037. Region: params.Region,
  6038. body: xml,
  6039. action: 'encryption',
  6040. headers: headers
  6041. }, function (err, data) {
  6042. if (err && err.statusCode === 204) {
  6043. return callback(null, { statusCode: err.statusCode });
  6044. } else if (err) {
  6045. return callback(err);
  6046. }
  6047. callback(null, {
  6048. statusCode: data.statusCode,
  6049. headers: data.headers
  6050. });
  6051. });
  6052. }
  6053. function getBucketEncryption(params, callback) {
  6054. submitRequest.call(this, {
  6055. Action: 'name/cos:GetBucketEncryption',
  6056. method: 'GET',
  6057. Bucket: params.Bucket,
  6058. Region: params.Region,
  6059. headers: params.Headers,
  6060. action: 'encryption'
  6061. }, function (err, data) {
  6062. if (err) {
  6063. if (err.statusCode === 404 && err.code === 'NoSuchEncryptionConfiguration') {
  6064. var result = {
  6065. EncryptionConfiguration: { Rules: [] },
  6066. statusCode: err.statusCode
  6067. };
  6068. err.headers && (result.headers = err.headers);
  6069. callback(null, result);
  6070. } else {
  6071. callback(err);
  6072. }
  6073. return;
  6074. }
  6075. var Rules = util.makeArray(data.EncryptionConfiguration && data.EncryptionConfiguration.Rule || []);
  6076. data.EncryptionConfiguration = { Rules: Rules };
  6077. callback(err, data);
  6078. });
  6079. }
  6080. function deleteBucketEncryption(params, callback) {
  6081. submitRequest.call(this, {
  6082. Action: 'name/cos:DeleteBucketReplication',
  6083. method: 'DELETE',
  6084. Bucket: params.Bucket,
  6085. Region: params.Region,
  6086. headers: params.Headers,
  6087. action: 'encryption'
  6088. }, function (err, data) {
  6089. if (err && err.statusCode === 204) {
  6090. return callback(null, { statusCode: err.statusCode });
  6091. } else if (err) {
  6092. return callback(err);
  6093. }
  6094. callback(null, {
  6095. statusCode: data.statusCode,
  6096. headers: data.headers
  6097. });
  6098. });
  6099. }
  6100. // Object 相关
  6101. /**
  6102. * 取回对应Object的元数据,Head的权限与Get的权限一致
  6103. * @param {Object} params 参数对象,必须
  6104. * @param {String} params.Bucket Bucket名称,必须
  6105. * @param {String} params.Region 地域名称,必须
  6106. * @param {String} params.Key 文件名称,必须
  6107. * @param {String} params.IfModifiedSince 当Object在指定时间后被修改,则返回对应Object元信息,否则返回304,非必须
  6108. * @param {Function} callback 回调函数,必须
  6109. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  6110. * @return {Object} data 为指定 object 的元数据,如果设置了 IfModifiedSince ,且文件未修改,则返回一个对象,NotModified 属性为 true
  6111. * @return {Boolean} data.NotModified 是否在 IfModifiedSince 时间点之后未修改该 object,则为 true
  6112. */
  6113. function headObject(params, callback) {
  6114. submitRequest.call(this, {
  6115. Action: 'name/cos:HeadObject',
  6116. method: 'HEAD',
  6117. Bucket: params.Bucket,
  6118. Region: params.Region,
  6119. Key: params.Key,
  6120. VersionId: params.VersionId,
  6121. headers: params.Headers
  6122. }, function (err, data) {
  6123. if (err) {
  6124. var statusCode = err.statusCode;
  6125. if (params.Headers['If-Modified-Since'] && statusCode && statusCode === 304) {
  6126. return callback(null, {
  6127. NotModified: true,
  6128. statusCode: statusCode
  6129. });
  6130. }
  6131. return callback(err);
  6132. }
  6133. data.ETag = util.attr(data.headers, 'etag', '');
  6134. callback(null, data);
  6135. });
  6136. }
  6137. function listObjectVersions(params, callback) {
  6138. var reqParams = {};
  6139. reqParams['prefix'] = params['Prefix'] || '';
  6140. reqParams['delimiter'] = params['Delimiter'];
  6141. reqParams['key-marker'] = params['KeyMarker'];
  6142. reqParams['version-id-marker'] = params['VersionIdMarker'];
  6143. reqParams['max-keys'] = params['MaxKeys'];
  6144. reqParams['encoding-type'] = params['EncodingType'];
  6145. submitRequest.call(this, {
  6146. Action: 'name/cos:GetBucketObjectVersions',
  6147. ResourceKey: reqParams['prefix'],
  6148. method: 'GET',
  6149. Bucket: params.Bucket,
  6150. Region: params.Region,
  6151. headers: params.Headers,
  6152. qs: reqParams,
  6153. action: 'versions'
  6154. }, function (err, data) {
  6155. if (err) return callback(err);
  6156. var ListVersionsResult = data.ListVersionsResult || {};
  6157. var DeleteMarkers = ListVersionsResult.DeleteMarker || [];
  6158. DeleteMarkers = util.isArray(DeleteMarkers) ? DeleteMarkers : [DeleteMarkers];
  6159. var Versions = ListVersionsResult.Version || [];
  6160. Versions = util.isArray(Versions) ? Versions : [Versions];
  6161. var result = util.clone(ListVersionsResult);
  6162. delete result.DeleteMarker;
  6163. delete result.Version;
  6164. util.extend(result, {
  6165. DeleteMarkers: DeleteMarkers,
  6166. Versions: Versions,
  6167. statusCode: data.statusCode,
  6168. headers: data.headers
  6169. });
  6170. callback(null, result);
  6171. });
  6172. }
  6173. /**
  6174. * 下载 object
  6175. * @param {Object} params 参数对象,必须
  6176. * @param {String} params.Bucket Bucket名称,必须
  6177. * @param {String} params.Region 地域名称,必须
  6178. * @param {String} params.Key 文件名称,必须
  6179. * @param {WriteStream} params.Output 文件写入流,非必须
  6180. * @param {String} params.IfModifiedSince 当Object在指定时间后被修改,则返回对应Object元信息,否则返回304,非必须
  6181. * @param {String} params.IfUnmodifiedSince 如果文件修改时间早于或等于指定时间,才返回文件内容。否则返回 412 (precondition failed),非必须
  6182. * @param {String} params.IfMatch 当 ETag 与指定的内容一致,才返回文件。否则返回 412 (precondition failed),非必须
  6183. * @param {String} params.IfNoneMatch 当 ETag 与指定的内容不一致,才返回文件。否则返回304 (not modified),非必须
  6184. * @param {String} params.ResponseContentType 设置返回头部中的 Content-Type 参数,非必须
  6185. * @param {String} params.ResponseContentLanguage 设置返回头部中的 Content-Language 参数,非必须
  6186. * @param {String} params.ResponseExpires 设置返回头部中的 Content-Expires 参数,非必须
  6187. * @param {String} params.ResponseCacheControl 设置返回头部中的 Cache-Control 参数,非必须
  6188. * @param {String} params.ResponseContentDisposition 设置返回头部中的 Content-Disposition 参数,非必须
  6189. * @param {String} params.ResponseContentEncoding 设置返回头部中的 Content-Encoding 参数,非必须
  6190. * @param {Function} callback 回调函数,必须
  6191. * @param {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  6192. * @param {Object} data 为对应的 object 数据,包括 body 和 headers
  6193. */
  6194. function getObject(params, callback) {
  6195. var reqParams = params.Query || {};
  6196. var reqParamsStr = params.QueryString || '';
  6197. var onProgress = util.throttleOnProgress.call(this, 0, params.onProgress);
  6198. reqParams['response-content-type'] = params['ResponseContentType'];
  6199. reqParams['response-content-language'] = params['ResponseContentLanguage'];
  6200. reqParams['response-expires'] = params['ResponseExpires'];
  6201. reqParams['response-cache-control'] = params['ResponseCacheControl'];
  6202. reqParams['response-content-disposition'] = params['ResponseContentDisposition'];
  6203. reqParams['response-content-encoding'] = params['ResponseContentEncoding'];
  6204. // 如果用户自己传入了 output
  6205. submitRequest.call(this, {
  6206. Action: 'name/cos:GetObject',
  6207. method: 'GET',
  6208. Bucket: params.Bucket,
  6209. Region: params.Region,
  6210. Key: params.Key,
  6211. VersionId: params.VersionId,
  6212. DataType: params.DataType,
  6213. headers: params.Headers,
  6214. qs: reqParams,
  6215. qsStr: reqParamsStr,
  6216. rawBody: true,
  6217. onDownloadProgress: onProgress
  6218. }, function (err, data) {
  6219. onProgress(null, true);
  6220. if (err) {
  6221. var statusCode = err.statusCode;
  6222. if (params.Headers['If-Modified-Since'] && statusCode && statusCode === 304) {
  6223. return callback(null, {
  6224. NotModified: true
  6225. });
  6226. }
  6227. return callback(err);
  6228. }
  6229. callback(null, {
  6230. Body: data.body,
  6231. ETag: util.attr(data.headers, 'etag', ''),
  6232. statusCode: data.statusCode,
  6233. headers: data.headers
  6234. });
  6235. });
  6236. }
  6237. /**
  6238. * 上传 object
  6239. * @param {Object} params 参数对象,必须
  6240. * @param {String} params.Bucket Bucket名称,必须
  6241. * @param {String} params.Region 地域名称,必须
  6242. * @param {String} params.Key 文件名称,必须
  6243. * @param {File || Blob || String} params.Body 上传文件对象或字符串,必须
  6244. * @param {String} params.CacheControl RFC 2616 中定义的缓存策略,将作为 Object 元数据保存,非必须
  6245. * @param {String} params.ContentDisposition RFC 2616 中定义的文件名称,将作为 Object 元数据保存,非必须
  6246. * @param {String} params.ContentEncoding RFC 2616 中定义的编码格式,将作为 Object 元数据保存,非必须
  6247. * @param {String} params.ContentLength RFC 2616 中定义的 HTTP 请求内容长度(字节),必须
  6248. * @param {String} params.ContentType RFC 2616 中定义的内容类型(MIME),将作为 Object 元数据保存,非必须
  6249. * @param {String} params.Expect 当使用 Expect: 100-continue 时,在收到服务端确认后,才会发送请求内容,非必须
  6250. * @param {String} params.Expires RFC 2616 中定义的过期时间,将作为 Object 元数据保存,非必须
  6251. * @param {String} params.ACL 允许用户自定义文件权限,有效值:private | public-read,非必须
  6252. * @param {String} params.GrantRead 赋予被授权者读取对象的权限,格式:id="[OwnerUin]",可使用半角逗号(,)分隔多组被授权者,非必须
  6253. * @param {String} params.GrantReadAcp 赋予被授权者读取对象的访问控制列表(ACL)的权限,格式:id="[OwnerUin]",可使用半角逗号(,)分隔多组被授权者,非必须
  6254. * @param {String} params.GrantWriteAcp 赋予被授权者写入对象的访问控制列表(ACL)的权限,格式:id="[OwnerUin]",可使用半角逗号(,)分隔多组被授权者,非必须
  6255. * @param {String} params.GrantFullControl 赋予被授权者操作对象的所有权限,格式:id="[OwnerUin]",可使用半角逗号(,)分隔多组被授权者,非必须
  6256. * @param {String} params.StorageClass 设置对象的存储级别,枚举值:STANDARD、STANDARD_IA、ARCHIVE,默认值:STANDARD,非必须
  6257. * @param {String} params.x-cos-meta-* 允许用户自定义的头部信息,将作为对象的元数据保存。大小限制2KB,非必须
  6258. * @param {String} params.ContentSha1 RFC 3174 中定义的 160-bit 内容 SHA-1 算法校验,非必须
  6259. * @param {String} params.ServerSideEncryption 支持按照指定的加密算法进行服务端数据加密,格式 x-cos-server-side-encryption: "AES256",非必须
  6260. * @param {Function} params.onProgress 上传进度回调函数
  6261. * @param {Function} callback 回调函数,必须
  6262. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  6263. * @return {Object} data 为对应的 object 数据
  6264. * @return {String} data.ETag 为对应上传文件的 ETag 值
  6265. */
  6266. function putObject(params, callback) {
  6267. var self = this;
  6268. var FileSize = params.ContentLength;
  6269. var onProgress = util.throttleOnProgress.call(self, FileSize, params.onProgress);
  6270. // 特殊处理 Cache-Control、Content-Type,避免代理更改这两个字段导致写入到 Object 属性里
  6271. var headers = params.Headers;
  6272. if (!headers['Cache-Control'] && !headers['cache-control']) headers['Cache-Control'] = '';
  6273. if (!headers['Content-Type'] && !headers['content-type']) headers['Content-Type'] = params.Body && params.Body.type || '';
  6274. var needCalcMd5 = params.UploadAddMetaMd5 || self.options.UploadAddMetaMd5 || self.options.UploadCheckContentMd5;
  6275. util.getBodyMd5(needCalcMd5, params.Body, function (md5) {
  6276. if (md5) {
  6277. if (self.options.UploadCheckContentMd5) headers['Content-MD5'] = util.binaryBase64(md5);
  6278. if (params.UploadAddMetaMd5 || self.options.UploadAddMetaMd5) headers['x-cos-meta-md5'] = md5;
  6279. }
  6280. if (params.ContentLength !== undefined) headers['Content-Length'] = params.ContentLength;
  6281. onProgress(null, true); // 任务状态开始 uploading
  6282. submitRequest.call(self, {
  6283. Action: 'name/cos:PutObject',
  6284. TaskId: params.TaskId,
  6285. method: 'PUT',
  6286. Bucket: params.Bucket,
  6287. Region: params.Region,
  6288. Key: params.Key,
  6289. headers: params.Headers,
  6290. qs: params.Query,
  6291. body: params.Body,
  6292. onProgress: onProgress
  6293. }, function (err, data) {
  6294. if (err) {
  6295. onProgress(null, true);
  6296. return callback(err);
  6297. }
  6298. onProgress({ loaded: FileSize, total: FileSize }, true);
  6299. var url = getUrl({
  6300. ForcePathStyle: self.options.ForcePathStyle,
  6301. protocol: self.options.Protocol,
  6302. domain: self.options.Domain,
  6303. bucket: params.Bucket,
  6304. region: !self.options.UseAccelerate ? params.Region : 'accelerate',
  6305. object: params.Key
  6306. });
  6307. url = url.substr(url.indexOf('://') + 3);
  6308. data.Location = url;
  6309. data.ETag = util.attr(data.headers, 'etag', '');
  6310. callback(null, data);
  6311. });
  6312. }, params.onHashProgress);
  6313. }
  6314. /**
  6315. * 删除 object
  6316. * @param {Object} params 参数对象,必须
  6317. * @param {String} params.Bucket Bucket名称,必须
  6318. * @param {String} params.Region 地域名称,必须
  6319. * @param {String} params.Key object名称,必须
  6320. * @param {Function} callback 回调函数,必须
  6321. * @param {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  6322. * @param {Object} data 删除操作成功之后返回的数据
  6323. */
  6324. function deleteObject(params, callback) {
  6325. submitRequest.call(this, {
  6326. Action: 'name/cos:DeleteObject',
  6327. method: 'DELETE',
  6328. Bucket: params.Bucket,
  6329. Region: params.Region,
  6330. Key: params.Key,
  6331. headers: params.Headers,
  6332. VersionId: params.VersionId
  6333. }, function (err, data) {
  6334. if (err) {
  6335. var statusCode = err.statusCode;
  6336. if (statusCode && statusCode === 404) {
  6337. return callback(null, { BucketNotFound: true, statusCode: statusCode });
  6338. } else {
  6339. return callback(err);
  6340. }
  6341. }
  6342. callback(null, {
  6343. statusCode: data.statusCode,
  6344. headers: data.headers
  6345. });
  6346. });
  6347. }
  6348. /**
  6349. * 获取 object 的 权限列表
  6350. * @param {Object} params 参数对象,必须
  6351. * @param {String} params.Bucket Bucket名称,必须
  6352. * @param {String} params.Region 地域名称,必须
  6353. * @param {String} params.Key object名称,必须
  6354. * @param {Function} callback 回调函数,必须
  6355. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  6356. * @return {Object} data 返回的数据
  6357. * @return {Object} data.AccessControlPolicy 权限列表
  6358. */
  6359. function getObjectAcl(params, callback) {
  6360. submitRequest.call(this, {
  6361. Action: 'name/cos:GetObjectACL',
  6362. method: 'GET',
  6363. Bucket: params.Bucket,
  6364. Region: params.Region,
  6365. Key: params.Key,
  6366. headers: params.Headers,
  6367. action: 'acl'
  6368. }, function (err, data) {
  6369. if (err) return callback(err);
  6370. var AccessControlPolicy = data.AccessControlPolicy || {};
  6371. var Owner = AccessControlPolicy.Owner || {};
  6372. var Grant = AccessControlPolicy.AccessControlList && AccessControlPolicy.AccessControlList.Grant || [];
  6373. Grant = util.isArray(Grant) ? Grant : [Grant];
  6374. var result = decodeAcl(AccessControlPolicy);
  6375. delete result.GrantWrite;
  6376. if (data.headers && data.headers['x-cos-acl']) {
  6377. result.ACL = data.headers['x-cos-acl'];
  6378. }
  6379. result = util.extend(result, {
  6380. Owner: Owner,
  6381. Grants: Grant,
  6382. statusCode: data.statusCode,
  6383. headers: data.headers
  6384. });
  6385. callback(null, result);
  6386. });
  6387. }
  6388. /**
  6389. * 设置 object 的 权限列表
  6390. * @param {Object} params 参数对象,必须
  6391. * @param {String} params.Bucket Bucket名称,必须
  6392. * @param {String} params.Region 地域名称,必须
  6393. * @param {String} params.Key object名称,必须
  6394. * @param {Function} callback 回调函数,必须
  6395. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  6396. * @return {Object} data 返回的数据
  6397. */
  6398. function putObjectAcl(params, callback) {
  6399. var headers = params.Headers;
  6400. var xml = '';
  6401. if (params['AccessControlPolicy']) {
  6402. var AccessControlPolicy = util.clone(params['AccessControlPolicy'] || {});
  6403. var Grants = AccessControlPolicy.Grants || AccessControlPolicy.Grant;
  6404. Grants = util.isArray(Grants) ? Grants : [Grants];
  6405. delete AccessControlPolicy.Grant;
  6406. delete AccessControlPolicy.Grants;
  6407. AccessControlPolicy.AccessControlList = { Grant: Grants };
  6408. xml = util.json2xml({ AccessControlPolicy: AccessControlPolicy });
  6409. headers['Content-Type'] = 'application/xml';
  6410. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  6411. }
  6412. // Grant Header 去重
  6413. util.each(headers, function (val, key) {
  6414. if (key.indexOf('x-cos-grant-') === 0) {
  6415. headers[key] = uniqGrant(headers[key]);
  6416. }
  6417. });
  6418. submitRequest.call(this, {
  6419. Action: 'name/cos:PutObjectACL',
  6420. method: 'PUT',
  6421. Bucket: params.Bucket,
  6422. Region: params.Region,
  6423. Key: params.Key,
  6424. action: 'acl',
  6425. headers: headers,
  6426. body: xml
  6427. }, function (err, data) {
  6428. if (err) return callback(err);
  6429. callback(null, {
  6430. statusCode: data.statusCode,
  6431. headers: data.headers
  6432. });
  6433. });
  6434. }
  6435. /**
  6436. * Options Object请求实现跨域访问的预请求。即发出一个 OPTIONS 请求给服务器以确认是否可以进行跨域操作。
  6437. * @param {Object} params 参数对象,必须
  6438. * @param {String} params.Bucket Bucket名称,必须
  6439. * @param {String} params.Region 地域名称,必须
  6440. * @param {String} params.Key object名称,必须
  6441. * @param {Function} callback 回调函数,必须
  6442. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  6443. * @return {Object} data 返回的数据
  6444. */
  6445. function optionsObject(params, callback) {
  6446. var headers = params.Headers;
  6447. headers['Origin'] = params['Origin'];
  6448. headers['Access-Control-Request-Method'] = params['AccessControlRequestMethod'];
  6449. headers['Access-Control-Request-Headers'] = params['AccessControlRequestHeaders'];
  6450. submitRequest.call(this, {
  6451. Action: 'name/cos:OptionsObject',
  6452. method: 'OPTIONS',
  6453. Bucket: params.Bucket,
  6454. Region: params.Region,
  6455. Key: params.Key,
  6456. headers: headers
  6457. }, function (err, data) {
  6458. if (err) {
  6459. if (err.statusCode && err.statusCode === 403) {
  6460. return callback(null, {
  6461. OptionsForbidden: true,
  6462. statusCode: err.statusCode
  6463. });
  6464. }
  6465. return callback(err);
  6466. }
  6467. var headers = data.headers || {};
  6468. callback(null, {
  6469. AccessControlAllowOrigin: headers['access-control-allow-origin'],
  6470. AccessControlAllowMethods: headers['access-control-allow-methods'],
  6471. AccessControlAllowHeaders: headers['access-control-allow-headers'],
  6472. AccessControlExposeHeaders: headers['access-control-expose-headers'],
  6473. AccessControlMaxAge: headers['access-control-max-age'],
  6474. statusCode: data.statusCode,
  6475. headers: data.headers
  6476. });
  6477. });
  6478. }
  6479. /**
  6480. * @param {Object} 参数列表
  6481. * @param {String} Bucket Bucket 名称
  6482. * @param {String} Region 地域名称
  6483. * @param {String} Key 文件名称
  6484. * @param {String} CopySource 源文件URL绝对路径,可以通过versionid子资源指定历史版本
  6485. * @param {String} ACL 允许用户自定义文件权限。有效值:private,public-read默认值:private。
  6486. * @param {String} GrantRead 赋予被授权者读的权限,格式 x-cos-grant-read: uin=" ",uin=" ",当需要给子账户授权时,uin="RootAcountID/SubAccountID",当需要给根账户授权时,uin="RootAcountID"。
  6487. * @param {String} GrantWrite 赋予被授权者写的权限,格式 x-cos-grant-write: uin=" ",uin=" ",当需要给子账户授权时,uin="RootAcountID/SubAccountID",当需要给根账户授权时,uin="RootAcountID"。
  6488. * @param {String} GrantFullControl 赋予被授权者读写权限,格式 x-cos-grant-full-control: uin=" ",uin=" ",当需要给子账户授权时,uin="RootAcountID/SubAccountID",当需要给根账户授权时,uin="RootAcountID"。
  6489. * @param {String} MetadataDirective 是否拷贝元数据,枚举值:Copy, Replaced,默认值Copy。假如标记为Copy,忽略Header中的用户元数据信息直接复制;假如标记为Replaced,按Header信息修改元数据。当目标路径和原路径一致,即用户试图修改元数据时,必须为Replaced
  6490. * @param {String} CopySourceIfModifiedSince 当Object在指定时间后被修改,则执行操作,否则返回412。可与x-cos-copy-source-If-None-Match一起使用,与其他条件联合使用返回冲突。
  6491. * @param {String} CopySourceIfUnmodifiedSince 当Object在指定时间后未被修改,则执行操作,否则返回412。可与x-cos-copy-source-If-Match一起使用,与其他条件联合使用返回冲突。
  6492. * @param {String} CopySourceIfMatch 当Object的ETag和给定一致时,则执行操作,否则返回412。可与x-cos-copy-source-If-Unmodified-Since一起使用,与其他条件联合使用返回冲突。
  6493. * @param {String} CopySourceIfNoneMatch 当Object的ETag和给定不一致时,则执行操作,否则返回412。可与x-cos-copy-source-If-Modified-Since一起使用,与其他条件联合使用返回冲突。
  6494. * @param {String} StorageClass 存储级别,枚举值:存储级别,枚举值:Standard, Standard_IA,Archive;默认值:Standard
  6495. * @param {String} CacheControl 指定所有缓存机制在整个请求/响应链中必须服从的指令。
  6496. * @param {String} ContentDisposition MIME 协议的扩展,MIME 协议指示 MIME 用户代理如何显示附加的文件
  6497. * @param {String} ContentEncoding HTTP 中用来对「采用何种编码格式传输正文」进行协定的一对头部字段
  6498. * @param {String} ContentLength 设置响应消息的实体内容的大小,单位为字节
  6499. * @param {String} ContentType RFC 2616 中定义的 HTTP 请求内容类型(MIME),例如text/plain
  6500. * @param {String} Expect 请求的特定的服务器行为
  6501. * @param {String} Expires 响应过期的日期和时间
  6502. * @param {String} params.ServerSideEncryption 支持按照指定的加密算法进行服务端数据加密,格式 x-cos-server-side-encryption: "AES256",非必须
  6503. * @param {String} ContentLanguage 指定内容语言
  6504. * @param {String} x-cos-meta-* 允许用户自定义的头部信息,将作为 Object 元数据返回。大小限制2K。
  6505. */
  6506. function putObjectCopy(params, callback) {
  6507. // 特殊处理 Cache-Control
  6508. var self = this;
  6509. var headers = params.Headers;
  6510. if (!headers['Cache-Control'] && !headers['cache-control']) headers['Cache-Control'] = '';
  6511. var CopySource = params.CopySource || '';
  6512. var m = CopySource.match(/^([^.]+-\d+)\.cos(v6)?\.([^.]+)\.[^/]+\/(.+)$/);
  6513. if (!m) {
  6514. callback(util.error(new Error('CopySource format error')));
  6515. return;
  6516. }
  6517. var SourceBucket = m[1];
  6518. var SourceRegion = m[3];
  6519. var SourceKey = decodeURIComponent(m[4]);
  6520. submitRequest.call(this, {
  6521. Scope: [{
  6522. action: 'name/cos:GetObject',
  6523. bucket: SourceBucket,
  6524. region: SourceRegion,
  6525. prefix: SourceKey
  6526. }, {
  6527. action: 'name/cos:PutObject',
  6528. bucket: params.Bucket,
  6529. region: params.Region,
  6530. prefix: params.Key
  6531. }],
  6532. method: 'PUT',
  6533. Bucket: params.Bucket,
  6534. Region: params.Region,
  6535. Key: params.Key,
  6536. VersionId: params.VersionId,
  6537. headers: params.Headers
  6538. }, function (err, data) {
  6539. if (err) return callback(err);
  6540. var result = util.clone(data.CopyObjectResult || {});
  6541. var url = getUrl({
  6542. ForcePathStyle: self.options.ForcePathStyle,
  6543. protocol: self.options.Protocol,
  6544. domain: self.options.Domain,
  6545. bucket: params.Bucket,
  6546. region: params.Region,
  6547. object: params.Key,
  6548. isLocation: true
  6549. });
  6550. util.extend(result, {
  6551. Location: url,
  6552. statusCode: data.statusCode,
  6553. headers: data.headers
  6554. });
  6555. callback(null, result);
  6556. });
  6557. }
  6558. function uploadPartCopy(params, callback) {
  6559. var CopySource = params.CopySource || '';
  6560. var m = CopySource.match(/^([^.]+-\d+)\.cos(v6)?\.([^.]+)\.[^/]+\/(.+)$/);
  6561. if (!m) {
  6562. callback(util.error(new Error('CopySource format error')));
  6563. return;
  6564. }
  6565. var SourceBucket = m[1];
  6566. var SourceRegion = m[3];
  6567. var SourceKey = decodeURIComponent(m[4]);
  6568. submitRequest.call(this, {
  6569. Scope: [{
  6570. action: 'name/cos:GetObject',
  6571. bucket: SourceBucket,
  6572. region: SourceRegion,
  6573. prefix: SourceKey
  6574. }, {
  6575. action: 'name/cos:PutObject',
  6576. bucket: params.Bucket,
  6577. region: params.Region,
  6578. prefix: params.Key
  6579. }],
  6580. method: 'PUT',
  6581. Bucket: params.Bucket,
  6582. Region: params.Region,
  6583. Key: params.Key,
  6584. VersionId: params.VersionId,
  6585. qs: {
  6586. partNumber: params['PartNumber'],
  6587. uploadId: params['UploadId']
  6588. },
  6589. headers: params.Headers
  6590. }, function (err, data) {
  6591. if (err) return callback(err);
  6592. var result = util.clone(data.CopyPartResult || {});
  6593. util.extend(result, {
  6594. statusCode: data.statusCode,
  6595. headers: data.headers
  6596. });
  6597. callback(null, result);
  6598. });
  6599. }
  6600. function deleteMultipleObject(params, callback) {
  6601. var Objects = params.Objects || [];
  6602. var Quiet = params.Quiet;
  6603. Objects = util.isArray(Objects) ? Objects : [Objects];
  6604. var xml = util.json2xml({ Delete: { Object: Objects, Quiet: Quiet || false } });
  6605. var headers = params.Headers;
  6606. headers['Content-Type'] = 'application/xml';
  6607. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  6608. var Scope = util.map(Objects, function (v) {
  6609. return {
  6610. action: 'name/cos:DeleteObject',
  6611. bucket: params.Bucket,
  6612. region: params.Region,
  6613. prefix: v.Key
  6614. };
  6615. });
  6616. submitRequest.call(this, {
  6617. Scope: Scope,
  6618. method: 'POST',
  6619. Bucket: params.Bucket,
  6620. Region: params.Region,
  6621. body: xml,
  6622. action: 'delete',
  6623. headers: headers
  6624. }, function (err, data) {
  6625. if (err) return callback(err);
  6626. var DeleteResult = data.DeleteResult || {};
  6627. var Deleted = DeleteResult.Deleted || [];
  6628. var Errors = DeleteResult.Error || [];
  6629. Deleted = util.isArray(Deleted) ? Deleted : [Deleted];
  6630. Errors = util.isArray(Errors) ? Errors : [Errors];
  6631. var result = util.clone(DeleteResult);
  6632. util.extend(result, {
  6633. Error: Errors,
  6634. Deleted: Deleted,
  6635. statusCode: data.statusCode,
  6636. headers: data.headers
  6637. });
  6638. callback(null, result);
  6639. });
  6640. }
  6641. function restoreObject(params, callback) {
  6642. var headers = params.Headers;
  6643. if (!params['RestoreRequest']) {
  6644. callback(util.error(new Error('missing param RestoreRequest')));
  6645. return;
  6646. }
  6647. var RestoreRequest = params.RestoreRequest || {};
  6648. var xml = util.json2xml({ RestoreRequest: RestoreRequest });
  6649. headers['Content-Type'] = 'application/xml';
  6650. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  6651. submitRequest.call(this, {
  6652. Action: 'name/cos:RestoreObject',
  6653. method: 'POST',
  6654. Bucket: params.Bucket,
  6655. Region: params.Region,
  6656. Key: params.Key,
  6657. VersionId: params.VersionId,
  6658. body: xml,
  6659. action: 'restore',
  6660. headers: headers
  6661. }, callback);
  6662. }
  6663. /**
  6664. * 设置 Object 的标签
  6665. * @param {Object} params 参数对象,必须
  6666. * @param {String} params.Bucket Object名称,必须
  6667. * @param {String} params.Region 地域名称,必须
  6668. * @param {Array} params.TagSet 标签设置,必须
  6669. * @param {Function} callback 回调函数,必须
  6670. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/42998
  6671. * @return {Object} data 返回数据
  6672. */
  6673. function putObjectTagging(params, callback) {
  6674. var Tagging = params['Tagging'] || {};
  6675. var Tags = Tagging.TagSet || Tagging.Tags || params['Tags'] || [];
  6676. Tags = util.clone(util.isArray(Tags) ? Tags : [Tags]);
  6677. var xml = util.json2xml({ Tagging: { TagSet: { Tag: Tags } } });
  6678. var headers = params.Headers;
  6679. headers['Content-Type'] = 'application/xml';
  6680. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  6681. submitRequest.call(this, {
  6682. Action: 'name/cos:PutObjectTagging',
  6683. method: 'PUT',
  6684. Bucket: params.Bucket,
  6685. Key: params.Key,
  6686. Region: params.Region,
  6687. body: xml,
  6688. action: 'tagging',
  6689. headers: headers,
  6690. VersionId: params.VersionId
  6691. }, function (err, data) {
  6692. if (err && err.statusCode === 204) {
  6693. return callback(null, { statusCode: err.statusCode });
  6694. } else if (err) {
  6695. return callback(err);
  6696. }
  6697. callback(null, {
  6698. statusCode: data.statusCode,
  6699. headers: data.headers
  6700. });
  6701. });
  6702. }
  6703. /**
  6704. * 获取 Object 的标签设置
  6705. * @param {Object} params 参数对象,必须
  6706. * @param {String} params.Bucket Bucket名称,必须
  6707. * @param {String} params.Region 地域名称,必须
  6708. * @param {Function} callback 回调函数,必须
  6709. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/42998
  6710. * @return {Object} data 返回数据
  6711. */
  6712. function getObjectTagging(params, callback) {
  6713. submitRequest.call(this, {
  6714. Action: 'name/cos:GetObjectTagging',
  6715. method: 'GET',
  6716. Key: params.Key,
  6717. Bucket: params.Bucket,
  6718. Region: params.Region,
  6719. headers: params.Headers,
  6720. action: 'tagging',
  6721. VersionId: params.VersionId
  6722. }, function (err, data) {
  6723. if (err) {
  6724. if (err.statusCode === 404 && err.error && (err.error === "Not Found" || err.error.Code === 'NoSuchTagSet')) {
  6725. var result = {
  6726. Tags: [],
  6727. statusCode: err.statusCode
  6728. };
  6729. err.headers && (result.headers = err.headers);
  6730. callback(null, result);
  6731. } else {
  6732. callback(err);
  6733. }
  6734. return;
  6735. }
  6736. var Tags = [];
  6737. try {
  6738. Tags = data.Tagging.TagSet.Tag || [];
  6739. } catch (e) {}
  6740. Tags = util.clone(util.isArray(Tags) ? Tags : [Tags]);
  6741. callback(null, {
  6742. Tags: Tags,
  6743. statusCode: data.statusCode,
  6744. headers: data.headers
  6745. });
  6746. });
  6747. }
  6748. /**
  6749. * 删除 Object 的 标签设置
  6750. * @param {Object} params 参数对象,必须
  6751. * @param {String} params.Bucket Object名称,必须
  6752. * @param {String} params.Region 地域名称,必须
  6753. * @param {Function} callback 回调函数,必须
  6754. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/42998
  6755. * @return {Object} data 返回的数据
  6756. */
  6757. function deleteObjectTagging(params, callback) {
  6758. submitRequest.call(this, {
  6759. Action: 'name/cos:DeleteObjectTagging',
  6760. method: 'DELETE',
  6761. Bucket: params.Bucket,
  6762. Region: params.Region,
  6763. Key: params.Key,
  6764. headers: params.Headers,
  6765. action: 'tagging',
  6766. VersionId: params.VersionId
  6767. }, function (err, data) {
  6768. if (err && err.statusCode === 204) {
  6769. return callback(null, { statusCode: err.statusCode });
  6770. } else if (err) {
  6771. return callback(err);
  6772. }
  6773. callback(null, {
  6774. statusCode: data.statusCode,
  6775. headers: data.headers
  6776. });
  6777. });
  6778. }
  6779. /**
  6780. * 使用 SQL 语句从指定对象(CSV 格式或者 JSON 格式)中检索内容
  6781. * @param {Object} params 参数对象,必须
  6782. * @param {String} params.Bucket Object名称,必须
  6783. * @param {String} params.Region 地域名称,必须
  6784. * @param {Object} params.SelectRequest 地域名称,必须
  6785. * @param {Function} callback 回调函数,必须
  6786. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/42998
  6787. * @return {Object} data 返回的数据
  6788. */
  6789. function selectObjectContent(params, callback) {
  6790. var SelectType = params['SelectType'];
  6791. if (!SelectType) return callback(util.error(new Error('missing param SelectType')));
  6792. var SelectRequest = params['SelectRequest'] || {};
  6793. var xml = util.json2xml({ SelectRequest: SelectRequest });
  6794. var headers = params.Headers;
  6795. headers['Content-Type'] = 'application/xml';
  6796. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  6797. submitRequest.call(this, {
  6798. Action: 'name/cos:GetObject',
  6799. method: 'POST',
  6800. Bucket: params.Bucket,
  6801. Region: params.Region,
  6802. Key: params.Key,
  6803. headers: params.Headers,
  6804. action: 'select',
  6805. qs: {
  6806. 'select-type': params['SelectType']
  6807. },
  6808. VersionId: params.VersionId,
  6809. body: xml,
  6810. DataType: 'arraybuffer',
  6811. rawBody: true
  6812. }, function (err, data) {
  6813. if (err && err.statusCode === 204) {
  6814. return callback(null, { statusCode: err.statusCode });
  6815. } else if (err) {
  6816. return callback(err);
  6817. }
  6818. var result = util.parseSelectPayload(data.body);
  6819. callback(null, {
  6820. statusCode: data.statusCode,
  6821. headers: data.headers,
  6822. Body: result.body,
  6823. Payload: result.payload
  6824. });
  6825. });
  6826. }
  6827. // 分块上传
  6828. /**
  6829. * 初始化分块上传
  6830. * @param {Object} params 参数对象,必须
  6831. * @param {String} params.Bucket Bucket名称,必须
  6832. * @param {String} params.Region 地域名称,必须
  6833. * @param {String} params.Key object名称,必须
  6834. * @param {String} params.UploadId object名称,必须
  6835. * @param {String} params.CacheControl RFC 2616 中定义的缓存策略,将作为 Object 元数据保存,非必须
  6836. * @param {String} params.ContentDisposition RFC 2616 中定义的文件名称,将作为 Object 元数据保存 ,非必须
  6837. * @param {String} params.ContentEncoding RFC 2616 中定义的编码格式,将作为 Object 元数据保存,非必须
  6838. * @param {String} params.ContentType RFC 2616 中定义的内容类型(MIME),将作为 Object 元数据保存,非必须
  6839. * @param {String} params.Expires RFC 2616 中定义的过期时间,将作为 Object 元数据保存,非必须
  6840. * @param {String} params.ACL 允许用户自定义文件权限,非必须
  6841. * @param {String} params.GrantRead 赋予被授权者读的权限 ,非必须
  6842. * @param {String} params.GrantWrite 赋予被授权者写的权限 ,非必须
  6843. * @param {String} params.GrantFullControl 赋予被授权者读写权限 ,非必须
  6844. * @param {String} params.StorageClass 设置Object的存储级别,枚举值:Standard,Standard_IA,Archive,非必须
  6845. * @param {String} params.ServerSideEncryption 支持按照指定的加密算法进行服务端数据加密,格式 x-cos-server-side-encryption: "AES256",非必须
  6846. * @param {Function} callback 回调函数,必须
  6847. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  6848. * @return {Object} data 返回的数据
  6849. */
  6850. function multipartInit(params, callback) {
  6851. var self = this;
  6852. // 特殊处理 Cache-Control
  6853. var headers = params.Headers;
  6854. // 特殊处理 Cache-Control、Content-Type
  6855. if (!headers['Cache-Control'] && !headers['cache-control']) headers['Cache-Control'] = '';
  6856. if (!headers['Content-Type'] && !headers['content-type']) headers['Content-Type'] = params.Body && params.Body.type || '';
  6857. util.getBodyMd5(params.Body && (params.UploadAddMetaMd5 || self.options.UploadAddMetaMd5), params.Body, function (md5) {
  6858. if (md5) params.Headers['x-cos-meta-md5'] = md5;
  6859. submitRequest.call(self, {
  6860. Action: 'name/cos:InitiateMultipartUpload',
  6861. method: 'POST',
  6862. Bucket: params.Bucket,
  6863. Region: params.Region,
  6864. Key: params.Key,
  6865. action: 'uploads',
  6866. headers: params.Headers,
  6867. qs: params.Query
  6868. }, function (err, data) {
  6869. if (err) return callback(err);
  6870. data = util.clone(data || {});
  6871. if (data && data.InitiateMultipartUploadResult) {
  6872. return callback(null, util.extend(data.InitiateMultipartUploadResult, {
  6873. statusCode: data.statusCode,
  6874. headers: data.headers
  6875. }));
  6876. }
  6877. callback(null, data);
  6878. });
  6879. }, params.onHashProgress);
  6880. }
  6881. /**
  6882. * 分块上传
  6883. * @param {Object} params 参数对象,必须
  6884. * @param {String} params.Bucket Bucket名称,必须
  6885. * @param {String} params.Region 地域名称,必须
  6886. * @param {String} params.Key object名称,必须
  6887. * @param {File || Blob || String} params.Body 上传文件对象或字符串
  6888. * @param {String} params.ContentLength RFC 2616 中定义的 HTTP 请求内容长度(字节),非必须
  6889. * @param {String} params.Expect 当使用 Expect: 100-continue 时,在收到服务端确认后,才会发送请求内容,非必须
  6890. * @param {String} params.ServerSideEncryption 支持按照指定的加密算法进行服务端数据加密,格式 x-cos-server-side-encryption: "AES256",非必须
  6891. * @param {String} params.ContentSha1 RFC 3174 中定义的 160-bit 内容 SHA-1 算法校验值,非必须
  6892. * @param {Function} callback 回调函数,必须
  6893. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  6894. * @return {Object} data 返回的数据
  6895. * @return {Object} data.ETag 返回的文件分块 sha1 值
  6896. */
  6897. function multipartUpload(params, callback) {
  6898. var self = this;
  6899. util.getFileSize('multipartUpload', params, function () {
  6900. util.getBodyMd5(self.options.UploadCheckContentMd5, params.Body, function (md5) {
  6901. if (md5) params.Headers['Content-MD5'] = util.binaryBase64(md5);
  6902. submitRequest.call(self, {
  6903. Action: 'name/cos:UploadPart',
  6904. TaskId: params.TaskId,
  6905. method: 'PUT',
  6906. Bucket: params.Bucket,
  6907. Region: params.Region,
  6908. Key: params.Key,
  6909. qs: {
  6910. partNumber: params['PartNumber'],
  6911. uploadId: params['UploadId']
  6912. },
  6913. headers: params.Headers,
  6914. onProgress: params.onProgress,
  6915. body: params.Body || null
  6916. }, function (err, data) {
  6917. if (err) return callback(err);
  6918. callback(null, {
  6919. ETag: util.attr(data.headers, 'etag', ''),
  6920. statusCode: data.statusCode,
  6921. headers: data.headers
  6922. });
  6923. });
  6924. });
  6925. });
  6926. }
  6927. /**
  6928. * 完成分块上传
  6929. * @param {Object} params 参数对象,必须
  6930. * @param {String} params.Bucket Bucket名称,必须
  6931. * @param {String} params.Region 地域名称,必须
  6932. * @param {String} params.Key object名称,必须
  6933. * @param {Array} params.Parts 分块信息列表,必须
  6934. * @param {String} params.Parts[i].PartNumber 块编号,必须
  6935. * @param {String} params.Parts[i].ETag 分块的 sha1 校验值
  6936. * @param {Function} callback 回调函数,必须
  6937. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  6938. * @return {Object} data 返回的数据
  6939. * @return {Object} data.CompleteMultipartUpload 完成分块上传后的文件信息,包括Location, Bucket, Key 和 ETag
  6940. */
  6941. function multipartComplete(params, callback) {
  6942. var self = this;
  6943. var UploadId = params.UploadId;
  6944. var Parts = params['Parts'];
  6945. for (var i = 0, len = Parts.length; i < len; i++) {
  6946. if (Parts[i]['ETag'].indexOf('"') === 0) {
  6947. continue;
  6948. }
  6949. Parts[i]['ETag'] = '"' + Parts[i]['ETag'] + '"';
  6950. }
  6951. var xml = util.json2xml({ CompleteMultipartUpload: { Part: Parts } });
  6952. // CSP/ceph CompleteMultipartUpload 接口 body 写死了限制 1MB,这里醉倒 10000 片时,xml 字符串去掉空格853KB
  6953. xml = xml.replace(/\n\s*/g, '');
  6954. var headers = params.Headers;
  6955. headers['Content-Type'] = 'application/xml';
  6956. headers['Content-MD5'] = util.binaryBase64(util.md5(xml));
  6957. submitRequest.call(this, {
  6958. Action: 'name/cos:CompleteMultipartUpload',
  6959. method: 'POST',
  6960. Bucket: params.Bucket,
  6961. Region: params.Region,
  6962. Key: params.Key,
  6963. qs: {
  6964. uploadId: UploadId
  6965. },
  6966. body: xml,
  6967. headers: headers
  6968. }, function (err, data) {
  6969. if (err) return callback(err);
  6970. var url = getUrl({
  6971. ForcePathStyle: self.options.ForcePathStyle,
  6972. protocol: self.options.Protocol,
  6973. domain: self.options.Domain,
  6974. bucket: params.Bucket,
  6975. region: params.Region,
  6976. object: params.Key,
  6977. isLocation: true
  6978. });
  6979. var res = data.CompleteMultipartUploadResult || {};
  6980. if (res.ProcessResults) {
  6981. if (res && res.ProcessResults) {
  6982. res.UploadResult = {
  6983. OriginalInfo: {
  6984. Key: res.Key,
  6985. Location: url,
  6986. ETag: res.ETag,
  6987. ImageInfo: res.ImageInfo
  6988. },
  6989. ProcessResults: res.ProcessResults
  6990. };
  6991. delete res.ImageInfo;
  6992. delete res.ProcessResults;
  6993. }
  6994. }
  6995. var result = util.extend(res, {
  6996. Location: url,
  6997. statusCode: data.statusCode,
  6998. headers: data.headers
  6999. });
  7000. callback(null, result);
  7001. });
  7002. }
  7003. /**
  7004. * 分块上传任务列表查询
  7005. * @param {Object} params 参数对象,必须
  7006. * @param {String} params.Bucket Bucket名称,必须
  7007. * @param {String} params.Region 地域名称,必须
  7008. * @param {String} params.Delimiter 定界符为一个符号,如果有Prefix,则将Prefix到delimiter之间的相同路径归为一类,定义为Common Prefix,然后列出所有Common Prefix。如果没有Prefix,则从路径起点开始,非必须
  7009. * @param {String} params.EncodingType 规定返回值的编码方式,非必须
  7010. * @param {String} params.Prefix 前缀匹配,用来规定返回的文件前缀地址,非必须
  7011. * @param {String} params.MaxUploads 单次返回最大的条目数量,默认1000,非必须
  7012. * @param {String} params.KeyMarker 与upload-id-marker一起使用 </Br>当upload-id-marker未被指定时,ObjectName字母顺序大于key-marker的条目将被列出 </Br>当upload-id-marker被指定时,ObjectName字母顺序大于key-marker的条目被列出,ObjectName字母顺序等于key-marker同时UploadId大于upload-id-marker的条目将被列出,非必须
  7013. * @param {String} params.UploadIdMarker 与key-marker一起使用 </Br>当key-marker未被指定时,upload-id-marker将被忽略 </Br>当key-marker被指定时,ObjectName字母顺序大于key-marker的条目被列出,ObjectName字母顺序等于key-marker同时UploadId大于upload-id-marker的条目将被列出,非必须
  7014. * @param {Function} callback 回调函数,必须
  7015. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  7016. * @return {Object} data 返回的数据
  7017. * @return {Object} data.ListMultipartUploadsResult 分块上传任务信息
  7018. */
  7019. function multipartList(params, callback) {
  7020. var reqParams = {};
  7021. reqParams['delimiter'] = params['Delimiter'];
  7022. reqParams['encoding-type'] = params['EncodingType'];
  7023. reqParams['prefix'] = params['Prefix'] || '';
  7024. reqParams['max-uploads'] = params['MaxUploads'];
  7025. reqParams['key-marker'] = params['KeyMarker'];
  7026. reqParams['upload-id-marker'] = params['UploadIdMarker'];
  7027. reqParams = util.clearKey(reqParams);
  7028. submitRequest.call(this, {
  7029. Action: 'name/cos:ListMultipartUploads',
  7030. ResourceKey: reqParams['prefix'],
  7031. method: 'GET',
  7032. Bucket: params.Bucket,
  7033. Region: params.Region,
  7034. headers: params.Headers,
  7035. qs: reqParams,
  7036. action: 'uploads'
  7037. }, function (err, data) {
  7038. if (err) return callback(err);
  7039. if (data && data.ListMultipartUploadsResult) {
  7040. var Upload = data.ListMultipartUploadsResult.Upload || [];
  7041. Upload = util.isArray(Upload) ? Upload : [Upload];
  7042. data.ListMultipartUploadsResult.Upload = Upload;
  7043. }
  7044. var result = util.clone(data.ListMultipartUploadsResult || {});
  7045. util.extend(result, {
  7046. statusCode: data.statusCode,
  7047. headers: data.headers
  7048. });
  7049. callback(null, result);
  7050. });
  7051. }
  7052. /**
  7053. * 上传的分块列表查询
  7054. * @param {Object} params 参数对象,必须
  7055. * @param {String} params.Bucket Bucket名称,必须
  7056. * @param {String} params.Region 地域名称,必须
  7057. * @param {String} params.Key object名称,必须
  7058. * @param {String} params.UploadId 标示本次分块上传的ID,必须
  7059. * @param {String} params.EncodingType 规定返回值的编码方式,非必须
  7060. * @param {String} params.MaxParts 单次返回最大的条目数量,默认1000,非必须
  7061. * @param {String} params.PartNumberMarker 默认以UTF-8二进制顺序列出条目,所有列出条目从marker开始,非必须
  7062. * @param {Function} callback 回调函数,必须
  7063. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  7064. * @return {Object} data 返回的数据
  7065. * @return {Object} data.ListMultipartUploadsResult 分块信息
  7066. */
  7067. function multipartListPart(params, callback) {
  7068. var reqParams = {};
  7069. reqParams['uploadId'] = params['UploadId'];
  7070. reqParams['encoding-type'] = params['EncodingType'];
  7071. reqParams['max-parts'] = params['MaxParts'];
  7072. reqParams['part-number-marker'] = params['PartNumberMarker'];
  7073. submitRequest.call(this, {
  7074. Action: 'name/cos:ListParts',
  7075. method: 'GET',
  7076. Bucket: params.Bucket,
  7077. Region: params.Region,
  7078. Key: params.Key,
  7079. headers: params.Headers,
  7080. qs: reqParams
  7081. }, function (err, data) {
  7082. if (err) return callback(err);
  7083. var ListPartsResult = data.ListPartsResult || {};
  7084. var Part = ListPartsResult.Part || [];
  7085. Part = util.isArray(Part) ? Part : [Part];
  7086. ListPartsResult.Part = Part;
  7087. var result = util.clone(ListPartsResult);
  7088. util.extend(result, {
  7089. statusCode: data.statusCode,
  7090. headers: data.headers
  7091. });
  7092. callback(null, result);
  7093. });
  7094. }
  7095. /**
  7096. * 抛弃分块上传
  7097. * @param {Object} params 参数对象,必须
  7098. * @param {String} params.Bucket Bucket名称,必须
  7099. * @param {String} params.Region 地域名称,必须
  7100. * @param {String} params.Key object名称,必须
  7101. * @param {String} params.UploadId 标示本次分块上传的ID,必须
  7102. * @param {Function} callback 回调函数,必须
  7103. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  7104. * @return {Object} data 返回的数据
  7105. */
  7106. function multipartAbort(params, callback) {
  7107. var reqParams = {};
  7108. reqParams['uploadId'] = params['UploadId'];
  7109. submitRequest.call(this, {
  7110. Action: 'name/cos:AbortMultipartUpload',
  7111. method: 'DELETE',
  7112. Bucket: params.Bucket,
  7113. Region: params.Region,
  7114. Key: params.Key,
  7115. headers: params.Headers,
  7116. qs: reqParams
  7117. }, function (err, data) {
  7118. if (err) return callback(err);
  7119. callback(null, {
  7120. statusCode: data.statusCode,
  7121. headers: data.headers
  7122. });
  7123. });
  7124. }
  7125. /**
  7126. * 抛弃分块上传
  7127. * @param {Object} params 参数对象,必须
  7128. * @param {String} params.Bucket Bucket名称,必须
  7129. * @param {String} params.Region 地域名称,必须
  7130. * @param {String} params.Key object名称,必须
  7131. * @param {String} params.UploadId 标示本次分块上传的ID,必须
  7132. * @param {Function} callback 回调函数,必须
  7133. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  7134. * @return {Object} data 返回的数据
  7135. */
  7136. function request(params, callback) {
  7137. submitRequest.call(this, {
  7138. method: params.Method,
  7139. Bucket: params.Bucket,
  7140. Region: params.Region,
  7141. Key: params.Key,
  7142. action: params.Action,
  7143. headers: params.Headers,
  7144. qs: params.Query,
  7145. body: params.Body
  7146. }, function (err, data) {
  7147. if (err) return callback(err);
  7148. if (data && data.body) {
  7149. data.Body = data.body;
  7150. delete data.body;
  7151. }
  7152. callback(err, data);
  7153. });
  7154. }
  7155. /**
  7156. * 获取签名
  7157. * @param {Object} params 参数对象,必须
  7158. * @param {String} params.Method 请求方法,必须
  7159. * @param {String} params.Key object名称,必须
  7160. * @param {String} params.Expires 名超时时间,单位秒,可选
  7161. * @return {String} data 返回签名字符串
  7162. */
  7163. function getAuth(params) {
  7164. var self = this;
  7165. return util.getAuth({
  7166. SecretId: params.SecretId || this.options.SecretId || '',
  7167. SecretKey: params.SecretKey || this.options.SecretKey || '',
  7168. Method: params.Method,
  7169. Key: params.Key,
  7170. Query: params.Query,
  7171. Headers: params.Headers,
  7172. Expires: params.Expires,
  7173. UseRawKey: self.options.UseRawKey,
  7174. SystemClockOffset: self.options.SystemClockOffset
  7175. });
  7176. }
  7177. /**
  7178. * 获取文件下载链接
  7179. * @param {Object} params 参数对象,必须
  7180. * @param {String} params.Bucket Bucket名称,必须
  7181. * @param {String} params.Region 地域名称,必须
  7182. * @param {String} params.Key object名称,必须
  7183. * @param {String} params.Method 请求的方法,可选
  7184. * @param {String} params.Expires 签名超时时间,单位秒,可选
  7185. * @param {Function} callback 回调函数,必须
  7186. * @return {Object} err 请求失败的错误,如果请求成功,则为空。https://cloud.tencent.com/document/product/436/7730
  7187. * @return {Object} data 返回的数据
  7188. */
  7189. function getObjectUrl(params, callback) {
  7190. var self = this;
  7191. var url = getUrl({
  7192. ForcePathStyle: self.options.ForcePathStyle,
  7193. protocol: params.Protocol || self.options.Protocol,
  7194. domain: params.Domain || self.options.Domain,
  7195. bucket: params.Bucket,
  7196. region: params.Region,
  7197. object: params.Key
  7198. });
  7199. var queryParamsStr = '';
  7200. if (params.Query) {
  7201. queryParamsStr += util.obj2str(params.Query);
  7202. }
  7203. if (params.QueryString) {
  7204. queryParamsStr += (queryParamsStr ? '&' : '') + params.QueryString;
  7205. }
  7206. var syncUrl = url;
  7207. if (params.Sign !== undefined && !params.Sign) {
  7208. queryParamsStr && (syncUrl += '?' + queryParamsStr);
  7209. callback(null, { Url: syncUrl });
  7210. return syncUrl;
  7211. }
  7212. var AuthData = getAuthorizationAsync.call(this, {
  7213. Action: (params.Method || '').toUpperCase() === 'PUT' ? 'name/cos:PutObject' : 'name/cos:GetObject',
  7214. Bucket: params.Bucket || '',
  7215. Region: params.Region || '',
  7216. Method: params.Method || 'get',
  7217. Key: params.Key,
  7218. Expires: params.Expires
  7219. }, function (err, AuthData) {
  7220. if (!callback) return;
  7221. if (err) {
  7222. callback(err);
  7223. return;
  7224. }
  7225. var signUrl = url;
  7226. signUrl += '?' + (AuthData.Authorization.indexOf('q-signature') > -1 ? AuthData.Authorization : 'sign=' + encodeURIComponent(AuthData.Authorization));
  7227. AuthData.SecurityToken && (signUrl += '&x-cos-security-token=' + AuthData.SecurityToken);
  7228. AuthData.ClientIP && (signUrl += '&clientIP=' + AuthData.ClientIP);
  7229. AuthData.ClientUA && (signUrl += '&clientUA=' + AuthData.ClientUA);
  7230. AuthData.Token && (signUrl += '&token=' + AuthData.Token);
  7231. queryParamsStr && (signUrl += '&' + queryParamsStr);
  7232. setTimeout(function () {
  7233. callback(null, { Url: signUrl });
  7234. });
  7235. });
  7236. if (AuthData) {
  7237. syncUrl += '?' + AuthData.Authorization + (AuthData.SecurityToken ? '&x-cos-security-token=' + AuthData.SecurityToken : '');
  7238. queryParamsStr && (syncUrl += '&' + queryParamsStr);
  7239. } else {
  7240. queryParamsStr && (syncUrl += '?' + queryParamsStr);
  7241. }
  7242. return syncUrl;
  7243. }
  7244. /**
  7245. * 私有方法
  7246. */
  7247. function decodeAcl(AccessControlPolicy) {
  7248. var result = {
  7249. GrantFullControl: [],
  7250. GrantWrite: [],
  7251. GrantRead: [],
  7252. GrantReadAcp: [],
  7253. GrantWriteAcp: [],
  7254. ACL: ''
  7255. };
  7256. var GrantMap = {
  7257. 'FULL_CONTROL': 'GrantFullControl',
  7258. 'WRITE': 'GrantWrite',
  7259. 'READ': 'GrantRead',
  7260. 'READ_ACP': 'GrantReadAcp',
  7261. 'WRITE_ACP': 'GrantWriteAcp'
  7262. };
  7263. var AccessControlList = AccessControlPolicy && AccessControlPolicy.AccessControlList || {};
  7264. var Grant = AccessControlList.Grant;
  7265. if (Grant) {
  7266. Grant = util.isArray(Grant) ? Grant : [Grant];
  7267. }
  7268. var PublicAcl = { READ: 0, WRITE: 0, FULL_CONTROL: 0 };
  7269. Grant && Grant.length && util.each(Grant, function (item) {
  7270. if (item.Grantee.ID === 'qcs::cam::anyone:anyone' || item.Grantee.URI === 'http://cam.qcloud.com/groups/global/AllUsers') {
  7271. PublicAcl[item.Permission] = 1;
  7272. } else if (item.Grantee.ID !== AccessControlPolicy.Owner.ID) {
  7273. result[GrantMap[item.Permission]].push('id="' + item.Grantee.ID + '"');
  7274. }
  7275. });
  7276. if (PublicAcl.FULL_CONTROL || PublicAcl.WRITE && PublicAcl.READ) {
  7277. result.ACL = 'public-read-write';
  7278. } else if (PublicAcl.READ) {
  7279. result.ACL = 'public-read';
  7280. } else {
  7281. result.ACL = 'private';
  7282. }
  7283. util.each(GrantMap, function (item) {
  7284. result[item] = uniqGrant(result[item].join(','));
  7285. });
  7286. return result;
  7287. }
  7288. // Grant 去重
  7289. function uniqGrant(str) {
  7290. var arr = str.split(',');
  7291. var exist = {};
  7292. var i, item;
  7293. for (i = 0; i < arr.length;) {
  7294. item = arr[i].trim();
  7295. if (exist[item]) {
  7296. arr.splice(i, 1);
  7297. } else {
  7298. exist[item] = true;
  7299. arr[i] = item;
  7300. i++;
  7301. }
  7302. }
  7303. return arr.join(',');
  7304. }
  7305. // 生成操作 url
  7306. function getUrl(params) {
  7307. var longBucket = params.bucket;
  7308. var shortBucket = longBucket.substr(0, longBucket.lastIndexOf('-'));
  7309. var appId = longBucket.substr(longBucket.lastIndexOf('-') + 1);
  7310. var domain = params.domain;
  7311. var region = params.region;
  7312. var object = params.object;
  7313. var protocol = params.protocol || (util.isBrowser && location.protocol === 'http:' ? 'http:' : 'https:');
  7314. if (!domain) {
  7315. if (['cn-south', 'cn-south-2', 'cn-north', 'cn-east', 'cn-southwest', 'sg'].indexOf(region) > -1) {
  7316. domain = '{Region}.myqcloud.com';
  7317. } else {
  7318. domain = 'cos.{Region}.myqcloud.com';
  7319. }
  7320. if (!params.ForcePathStyle) {
  7321. domain = '{Bucket}.' + domain;
  7322. }
  7323. }
  7324. domain = domain.replace(/\{\{AppId\}\}/ig, appId).replace(/\{\{Bucket\}\}/ig, shortBucket).replace(/\{\{Region\}\}/ig, region).replace(/\{\{.*?\}\}/ig, '');
  7325. domain = domain.replace(/\{AppId\}/ig, appId).replace(/\{BucketName\}/ig, shortBucket).replace(/\{Bucket\}/ig, longBucket).replace(/\{Region\}/ig, region).replace(/\{.*?\}/ig, '');
  7326. if (!/^[a-zA-Z]+:\/\//.test(domain)) {
  7327. domain = protocol + '//' + domain;
  7328. }
  7329. // 去掉域名最后的斜杆
  7330. if (domain.slice(-1) === '/') {
  7331. domain = domain.slice(0, -1);
  7332. }
  7333. var url = domain;
  7334. if (params.ForcePathStyle) {
  7335. url += '/' + longBucket;
  7336. }
  7337. url += '/';
  7338. if (object) {
  7339. url += util.camSafeUrlEncode(object).replace(/%2F/g, '/');
  7340. }
  7341. if (params.isLocation) {
  7342. url = url.replace(/^https?:\/\//, '');
  7343. }
  7344. return url;
  7345. }
  7346. // 异步获取签名
  7347. function getAuthorizationAsync(params, callback) {
  7348. var headers = util.clone(params.Headers);
  7349. util.each(headers, function (v, k) {
  7350. (v === '' || ['content-type', 'cache-control', 'expires'].indexOf(k.toLowerCase()) > -1) && delete headers[k];
  7351. });
  7352. // 获取凭证的回调,避免用户 callback 多次
  7353. var cbDone = false;
  7354. var cb = function (err, AuthData) {
  7355. if (cbDone) return;
  7356. cbDone = true;
  7357. if (AuthData && AuthData.XCosSecurityToken && !AuthData.SecurityToken) {
  7358. AuthData = util.clone(AuthData);
  7359. AuthData.SecurityToken = AuthData.XCosSecurityToken;
  7360. delete AuthData.XCosSecurityToken;
  7361. }
  7362. callback && callback(err, AuthData);
  7363. };
  7364. var self = this;
  7365. var Bucket = params.Bucket || '';
  7366. var Region = params.Region || '';
  7367. // PathName
  7368. var KeyName = params.Key || '';
  7369. if (self.options.ForcePathStyle && Bucket) {
  7370. KeyName = Bucket + '/' + KeyName;
  7371. }
  7372. var Pathname = '/' + KeyName;
  7373. // Action、ResourceKey
  7374. var StsData = {};
  7375. var Scope = params.Scope;
  7376. if (!Scope) {
  7377. var Action = params.Action || '';
  7378. var ResourceKey = params.ResourceKey || params.Key || '';
  7379. Scope = params.Scope || [{
  7380. action: Action,
  7381. bucket: Bucket,
  7382. region: Region,
  7383. prefix: ResourceKey
  7384. }];
  7385. }
  7386. var ScopeKey = util.md5(JSON.stringify(Scope));
  7387. // STS
  7388. self._StsCache = self._StsCache || [];
  7389. (function () {
  7390. var i, AuthData;
  7391. for (i = self._StsCache.length - 1; i >= 0; i--) {
  7392. AuthData = self._StsCache[i];
  7393. var compareTime = Math.round(util.getSkewTime(self.options.SystemClockOffset) / 1000) + 30;
  7394. if (AuthData.StartTime && compareTime < AuthData.StartTime || compareTime >= AuthData.ExpiredTime) {
  7395. self._StsCache.splice(i, 1);
  7396. continue;
  7397. }
  7398. if (!AuthData.ScopeLimit || AuthData.ScopeLimit && AuthData.ScopeKey === ScopeKey) {
  7399. StsData = AuthData;
  7400. break;
  7401. }
  7402. }
  7403. })();
  7404. var calcAuthByTmpKey = function () {
  7405. var KeyTime = StsData.StartTime && StsData.ExpiredTime ? StsData.StartTime + ';' + StsData.ExpiredTime : '';
  7406. var Authorization = util.getAuth({
  7407. SecretId: StsData.TmpSecretId,
  7408. SecretKey: StsData.TmpSecretKey,
  7409. Method: params.Method,
  7410. Pathname: Pathname,
  7411. Query: params.Query,
  7412. Headers: headers,
  7413. Expires: params.Expires,
  7414. UseRawKey: self.options.UseRawKey,
  7415. SystemClockOffset: self.options.SystemClockOffset,
  7416. KeyTime: KeyTime
  7417. });
  7418. var AuthData = {
  7419. Authorization: Authorization,
  7420. SecurityToken: StsData.SecurityToken || StsData.XCosSecurityToken || '',
  7421. Token: StsData.Token || '',
  7422. ClientIP: StsData.ClientIP || '',
  7423. ClientUA: StsData.ClientUA || ''
  7424. };
  7425. cb(null, AuthData);
  7426. };
  7427. var checkAuthError = function (AuthData) {
  7428. if (AuthData.Authorization) {
  7429. // 检查签名格式
  7430. var formatAllow = false;
  7431. var auth = AuthData.Authorization;
  7432. if (auth) {
  7433. if (auth.indexOf(' ') > -1) {
  7434. formatAllow = false;
  7435. } else if (auth.indexOf('q-sign-algorithm=') > -1 && auth.indexOf('q-ak=') > -1 && auth.indexOf('q-sign-time=') > -1 && auth.indexOf('q-key-time=') > -1 && auth.indexOf('q-url-param-list=') > -1) {
  7436. formatAllow = true;
  7437. } else {
  7438. try {
  7439. auth = atob(auth);
  7440. if (auth.indexOf('a=') > -1 && auth.indexOf('k=') > -1 && auth.indexOf('t=') > -1 && auth.indexOf('r=') > -1 && auth.indexOf('b=') > -1) {
  7441. formatAllow = true;
  7442. }
  7443. } catch (e) {}
  7444. }
  7445. }
  7446. if (!formatAllow) return util.error(new Error('getAuthorization callback params format error'));
  7447. } else {
  7448. if (!AuthData.TmpSecretId) return util.error(new Error('getAuthorization callback params missing "TmpSecretId"'));
  7449. if (!AuthData.TmpSecretKey) return util.error(new Error('getAuthorization callback params missing "TmpSecretKey"'));
  7450. if (!AuthData.SecurityToken && !AuthData.XCosSecurityToken) return util.error(new Error('getAuthorization callback params missing "SecurityToken"'));
  7451. if (!AuthData.ExpiredTime) return util.error(new Error('getAuthorization callback params missing "ExpiredTime"'));
  7452. if (AuthData.ExpiredTime && AuthData.ExpiredTime.toString().length !== 10) return util.error(new Error('getAuthorization callback params "ExpiredTime" should be 10 digits'));
  7453. if (AuthData.StartTime && AuthData.StartTime.toString().length !== 10) return util.error(new Error('getAuthorization callback params "StartTime" should be 10 StartTime'));
  7454. }
  7455. return false;
  7456. };
  7457. // 先判断是否有临时密钥
  7458. if (StsData.ExpiredTime && StsData.ExpiredTime - util.getSkewTime(self.options.SystemClockOffset) / 1000 > 60) {
  7459. // 如果缓存的临时密钥有效,并还有超过60秒有效期就直接使用
  7460. calcAuthByTmpKey();
  7461. } else if (self.options.getAuthorization) {
  7462. // 外部计算签名或获取临时密钥
  7463. self.options.getAuthorization.call(self, {
  7464. Bucket: Bucket,
  7465. Region: Region,
  7466. Method: params.Method,
  7467. Key: KeyName,
  7468. Pathname: Pathname,
  7469. Query: params.Query,
  7470. Headers: headers,
  7471. Scope: Scope,
  7472. SystemClockOffset: self.options.SystemClockOffset
  7473. }, function (AuthData) {
  7474. if (typeof AuthData === 'string') AuthData = { Authorization: AuthData };
  7475. var AuthError = checkAuthError(AuthData);
  7476. if (AuthError) return cb(AuthError);
  7477. if (AuthData.Authorization) {
  7478. cb(null, AuthData);
  7479. } else {
  7480. StsData = AuthData || {};
  7481. StsData.Scope = Scope;
  7482. StsData.ScopeKey = ScopeKey;
  7483. self._StsCache.push(StsData);
  7484. calcAuthByTmpKey();
  7485. }
  7486. });
  7487. } else if (self.options.getSTS) {
  7488. // 外部获取临时密钥
  7489. self.options.getSTS.call(self, {
  7490. Bucket: Bucket,
  7491. Region: Region
  7492. }, function (data) {
  7493. StsData = data || {};
  7494. StsData.Scope = Scope;
  7495. StsData.ScopeKey = ScopeKey;
  7496. if (!StsData.TmpSecretId) StsData.TmpSecretId = StsData.SecretId;
  7497. if (!StsData.TmpSecretKey) StsData.TmpSecretKey = StsData.SecretKey;
  7498. var AuthError = checkAuthError(StsData);
  7499. if (AuthError) return cb(AuthError);
  7500. self._StsCache.push(StsData);
  7501. calcAuthByTmpKey();
  7502. });
  7503. } else {
  7504. // 内部计算获取签名
  7505. return function () {
  7506. var Authorization = util.getAuth({
  7507. SecretId: params.SecretId || self.options.SecretId,
  7508. SecretKey: params.SecretKey || self.options.SecretKey,
  7509. Method: params.Method,
  7510. Pathname: Pathname,
  7511. Query: params.Query,
  7512. Headers: headers,
  7513. Expires: params.Expires,
  7514. UseRawKey: self.options.UseRawKey,
  7515. SystemClockOffset: self.options.SystemClockOffset
  7516. });
  7517. var AuthData = {
  7518. Authorization: Authorization,
  7519. SecurityToken: self.options.SecurityToken || self.options.XCosSecurityToken
  7520. };
  7521. cb(null, AuthData);
  7522. return AuthData;
  7523. }();
  7524. }
  7525. return '';
  7526. }
  7527. // 调整时间偏差
  7528. function allowRetry(err) {
  7529. var allowRetry = false;
  7530. var isTimeError = false;
  7531. var serverDate = err.headers && (err.headers.date || err.headers.Date) || err.error && err.error.ServerTime;
  7532. try {
  7533. var errorCode = err.error.Code;
  7534. var errorMessage = err.error.Message;
  7535. if (errorCode === 'RequestTimeTooSkewed' || errorCode === 'AccessDenied' && errorMessage === 'Request has expired') {
  7536. isTimeError = true;
  7537. }
  7538. } catch (e) {}
  7539. if (err) {
  7540. if (isTimeError && serverDate) {
  7541. var serverTime = Date.parse(serverDate);
  7542. if (this.options.CorrectClockSkew && Math.abs(util.getSkewTime(this.options.SystemClockOffset) - serverTime) >= 30000) {
  7543. console.error('error: Local time is too skewed.');
  7544. this.options.SystemClockOffset = serverTime - Date.now();
  7545. allowRetry = true;
  7546. }
  7547. } else if (Math.floor(err.statusCode / 100) === 5) {
  7548. allowRetry = true;
  7549. }
  7550. }
  7551. return allowRetry;
  7552. }
  7553. // 获取签名并发起请求
  7554. function submitRequest(params, callback) {
  7555. var self = this;
  7556. // 处理 headers
  7557. !params.headers && (params.headers = {});
  7558. // 处理 query
  7559. !params.qs && (params.qs = {});
  7560. params.VersionId && (params.qs.versionId = params.VersionId);
  7561. params.qs = util.clearKey(params.qs);
  7562. // 清理 undefined 和 null 字段
  7563. params.headers && (params.headers = util.clearKey(params.headers));
  7564. params.qs && (params.qs = util.clearKey(params.qs));
  7565. var Query = util.clone(params.qs);
  7566. params.action && (Query[params.action] = '');
  7567. var next = function (tryTimes) {
  7568. var oldClockOffset = self.options.SystemClockOffset;
  7569. getAuthorizationAsync.call(self, {
  7570. Bucket: params.Bucket || '',
  7571. Region: params.Region || '',
  7572. Method: params.method,
  7573. Key: params.Key,
  7574. Query: Query,
  7575. Headers: params.headers,
  7576. Action: params.Action,
  7577. ResourceKey: params.ResourceKey,
  7578. Scope: params.Scope
  7579. }, function (err, AuthData) {
  7580. if (err) {
  7581. callback(err);
  7582. return;
  7583. }
  7584. params.AuthData = AuthData;
  7585. _submitRequest.call(self, params, function (err, data) {
  7586. if (err && tryTimes < 2 && (oldClockOffset !== self.options.SystemClockOffset || allowRetry.call(self, err))) {
  7587. if (params.headers) {
  7588. delete params.headers.Authorization;
  7589. delete params.headers['token'];
  7590. delete params.headers['clientIP'];
  7591. delete params.headers['clientUA'];
  7592. delete params.headers['x-cos-security-token'];
  7593. }
  7594. next(tryTimes + 1);
  7595. } else {
  7596. callback(err, data);
  7597. }
  7598. });
  7599. });
  7600. };
  7601. next(1);
  7602. }
  7603. // 发起请求
  7604. function _submitRequest(params, callback) {
  7605. var self = this;
  7606. var TaskId = params.TaskId;
  7607. if (TaskId && !self._isRunningTask(TaskId)) return;
  7608. var bucket = params.Bucket;
  7609. var region = params.Region;
  7610. var object = params.Key;
  7611. var method = params.method || 'GET';
  7612. var url = params.url;
  7613. var body = params.body;
  7614. var rawBody = params.rawBody;
  7615. // url
  7616. if (self.options.UseAccelerate) {
  7617. region = 'accelerate';
  7618. }
  7619. url = url || getUrl({
  7620. ForcePathStyle: self.options.ForcePathStyle,
  7621. protocol: self.options.Protocol,
  7622. domain: self.options.Domain,
  7623. bucket: bucket,
  7624. region: region,
  7625. object: object
  7626. });
  7627. if (params.action) {
  7628. url = url + '?' + params.action;
  7629. }
  7630. if (params.qsStr) {
  7631. if (url.indexOf('?') > -1) {
  7632. url = url + '&' + params.qsStr;
  7633. } else {
  7634. url = url + '?' + params.qsStr;
  7635. }
  7636. }
  7637. var opt = {
  7638. method: method,
  7639. url: url,
  7640. headers: params.headers,
  7641. qs: params.qs,
  7642. body: body
  7643. };
  7644. // 获取签名
  7645. opt.headers.Authorization = params.AuthData.Authorization;
  7646. params.AuthData.Token && (opt.headers['token'] = params.AuthData.Token);
  7647. params.AuthData.ClientIP && (opt.headers['clientIP'] = params.AuthData.ClientIP);
  7648. params.AuthData.ClientUA && (opt.headers['clientUA'] = params.AuthData.ClientUA);
  7649. params.AuthData.SecurityToken && (opt.headers['x-cos-security-token'] = params.AuthData.SecurityToken);
  7650. // 清理 undefined 和 null 字段
  7651. opt.headers && (opt.headers = util.clearKey(opt.headers));
  7652. opt = util.clearKey(opt);
  7653. // progress
  7654. if (params.onProgress && typeof params.onProgress === 'function') {
  7655. var contentLength = body && (body.size || body.length) || 0;
  7656. opt.onProgress = function (e) {
  7657. if (TaskId && !self._isRunningTask(TaskId)) return;
  7658. var loaded = e ? e.loaded : 0;
  7659. params.onProgress({ loaded: loaded, total: contentLength });
  7660. };
  7661. }
  7662. if (params.onDownloadProgress) {
  7663. opt.onDownloadProgress = params.onDownloadProgress;
  7664. }
  7665. if (params.DataType) {
  7666. opt.dataType = params.DataType;
  7667. }
  7668. if (this.options.Timeout) {
  7669. opt.timeout = this.options.Timeout;
  7670. }
  7671. self.options.ForcePathStyle && (opt.pathStyle = self.options.ForcePathStyle);
  7672. self.emit('before-send', opt);
  7673. var sender = (self.options.Request || REQUEST)(opt, function (r) {
  7674. if (r.error === 'abort') return;
  7675. // 抛出事件,允许修改返回值的 error、statusCode、statusMessage、body
  7676. self.emit('after-receive', r);
  7677. var response = { statusCode: r.statusCode, statusMessage: r.statusMessage, headers: r.headers };
  7678. var err = r.error;
  7679. var body = r.body;
  7680. // 返回内容添加 状态码 和 headers
  7681. var hasReturned;
  7682. var cb = function (err, data) {
  7683. TaskId && self.off('inner-kill-task', killTask);
  7684. if (hasReturned) return;
  7685. hasReturned = true;
  7686. var attrs = {};
  7687. response && response.statusCode && (attrs.statusCode = response.statusCode);
  7688. response && response.headers && (attrs.headers = response.headers);
  7689. if (err) {
  7690. err = util.extend(err || {}, attrs);
  7691. callback(err, null);
  7692. } else {
  7693. data = util.extend(data || {}, attrs);
  7694. callback(null, data);
  7695. }
  7696. sender = null;
  7697. };
  7698. // 请求错误,发生网络错误
  7699. if (err) return cb(util.error(err));
  7700. // 请求返回码不为 200
  7701. var statusCode = response.statusCode;
  7702. var statusSuccess = Math.floor(statusCode / 100) === 2; // 200 202 204 206
  7703. // 不对 body 进行转换,body 直接挂载返回
  7704. if (rawBody && statusSuccess) return cb(null, { body: body });
  7705. // 解析 xml body
  7706. var json;
  7707. try {
  7708. json = body && body.indexOf('<') > -1 && body.indexOf('>') > -1 && util.xml2json(body) || {};
  7709. } catch (e) {
  7710. json = {};
  7711. }
  7712. // 处理返回值
  7713. var xmlError = json && json.Error;
  7714. if (statusSuccess) {
  7715. // 正确返回,状态码 2xx 时,body 不会有 Error
  7716. cb(null, json);
  7717. } else if (xmlError) {
  7718. // 正常返回了 xml body,且有 Error 节点
  7719. cb(util.error(new Error(xmlError.Message), { code: xmlError.Code, error: xmlError }));
  7720. } else if (statusCode) {
  7721. // 有错误的状态码
  7722. cb(util.error(new Error(response.statusMessage), { code: '' + statusCode }));
  7723. } else if (statusCode) {
  7724. // 无状态码,或者获取不到状态码
  7725. cb(util.error(new Error('statusCode error')));
  7726. }
  7727. });
  7728. // kill task
  7729. var killTask = function (data) {
  7730. if (data.TaskId === TaskId) {
  7731. sender && sender.abort && sender.abort();
  7732. self.off('inner-kill-task', killTask);
  7733. }
  7734. };
  7735. TaskId && self.on('inner-kill-task', killTask);
  7736. }
  7737. var API_MAP = {
  7738. // Bucket 相关方法
  7739. getService: getService, // Bucket
  7740. putBucket: putBucket,
  7741. headBucket: headBucket, // Bucket
  7742. getBucket: getBucket,
  7743. deleteBucket: deleteBucket,
  7744. putBucketAcl: putBucketAcl, // BucketACL
  7745. getBucketAcl: getBucketAcl,
  7746. putBucketCors: putBucketCors, // BucketCors
  7747. getBucketCors: getBucketCors,
  7748. deleteBucketCors: deleteBucketCors,
  7749. getBucketLocation: getBucketLocation, // BucketLocation
  7750. getBucketPolicy: getBucketPolicy, // BucketPolicy
  7751. putBucketPolicy: putBucketPolicy,
  7752. deleteBucketPolicy: deleteBucketPolicy,
  7753. putBucketTagging: putBucketTagging, // BucketTagging
  7754. getBucketTagging: getBucketTagging,
  7755. deleteBucketTagging: deleteBucketTagging,
  7756. putBucketLifecycle: putBucketLifecycle, // BucketLifecycle
  7757. getBucketLifecycle: getBucketLifecycle,
  7758. deleteBucketLifecycle: deleteBucketLifecycle,
  7759. putBucketVersioning: putBucketVersioning, // BucketVersioning
  7760. getBucketVersioning: getBucketVersioning,
  7761. putBucketReplication: putBucketReplication, // BucketReplication
  7762. getBucketReplication: getBucketReplication,
  7763. deleteBucketReplication: deleteBucketReplication,
  7764. putBucketWebsite: putBucketWebsite, // BucketWebsite
  7765. getBucketWebsite: getBucketWebsite,
  7766. deleteBucketWebsite: deleteBucketWebsite,
  7767. putBucketReferer: putBucketReferer, // BucketReferer
  7768. getBucketReferer: getBucketReferer,
  7769. putBucketDomain: putBucketDomain, // BucketDomain
  7770. getBucketDomain: getBucketDomain,
  7771. deleteBucketDomain: deleteBucketDomain,
  7772. putBucketOrigin: putBucketOrigin, // BucketOrigin
  7773. getBucketOrigin: getBucketOrigin,
  7774. deleteBucketOrigin: deleteBucketOrigin,
  7775. putBucketLogging: putBucketLogging, // BucketLogging
  7776. getBucketLogging: getBucketLogging,
  7777. putBucketInventory: putBucketInventory, // BucketInventory
  7778. getBucketInventory: getBucketInventory,
  7779. listBucketInventory: listBucketInventory,
  7780. deleteBucketInventory: deleteBucketInventory,
  7781. putBucketAccelerate: putBucketAccelerate,
  7782. getBucketAccelerate: getBucketAccelerate,
  7783. putBucketEncryption: putBucketEncryption,
  7784. getBucketEncryption: getBucketEncryption,
  7785. deleteBucketEncryption: deleteBucketEncryption,
  7786. // Object 相关方法
  7787. getObject: getObject,
  7788. headObject: headObject,
  7789. listObjectVersions: listObjectVersions,
  7790. putObject: putObject,
  7791. deleteObject: deleteObject,
  7792. getObjectAcl: getObjectAcl,
  7793. putObjectAcl: putObjectAcl,
  7794. optionsObject: optionsObject,
  7795. putObjectCopy: putObjectCopy,
  7796. deleteMultipleObject: deleteMultipleObject,
  7797. restoreObject: restoreObject,
  7798. putObjectTagging: putObjectTagging,
  7799. getObjectTagging: getObjectTagging,
  7800. deleteObjectTagging: deleteObjectTagging,
  7801. selectObjectContent: selectObjectContent,
  7802. // 分块上传相关方法
  7803. uploadPartCopy: uploadPartCopy,
  7804. multipartInit: multipartInit,
  7805. multipartUpload: multipartUpload,
  7806. multipartComplete: multipartComplete,
  7807. multipartList: multipartList,
  7808. multipartListPart: multipartListPart,
  7809. multipartAbort: multipartAbort,
  7810. // 工具方法
  7811. request: request,
  7812. getObjectUrl: getObjectUrl,
  7813. getAuth: getAuth
  7814. };
  7815. function warnOldApi(apiName, fn, proto) {
  7816. util.each(['Cors', 'Acl'], function (suffix) {
  7817. if (apiName.slice(-suffix.length) === suffix) {
  7818. var oldName = apiName.slice(0, -suffix.length) + suffix.toUpperCase();
  7819. var apiFn = util.apiWrapper(apiName, fn);
  7820. var warned = false;
  7821. proto[oldName] = function () {
  7822. !warned && console.warn('warning: cos.' + oldName + ' has been deprecated. Please Use cos.' + apiName + ' instead.');
  7823. warned = true;
  7824. apiFn.apply(this, arguments);
  7825. };
  7826. }
  7827. });
  7828. }
  7829. module.exports.init = function (COS, task) {
  7830. task.transferToTaskMethod(API_MAP, 'putObject');
  7831. util.each(API_MAP, function (fn, apiName) {
  7832. COS.prototype[apiName] = util.apiWrapper(apiName, fn);
  7833. warnOldApi(apiName, fn, COS.prototype);
  7834. });
  7835. };
  7836. /***/ }),
  7837. /* 17 */
  7838. /***/ (function(module, exports) {
  7839. var stringifyPrimitive = function (v) {
  7840. switch (typeof v) {
  7841. case 'string':
  7842. return v;
  7843. case 'boolean':
  7844. return v ? 'true' : 'false';
  7845. case 'number':
  7846. return isFinite(v) ? v : '';
  7847. default:
  7848. return '';
  7849. }
  7850. };
  7851. var queryStringify = function (obj, sep, eq, name) {
  7852. sep = sep || '&';
  7853. eq = eq || '=';
  7854. if (obj === null) {
  7855. obj = undefined;
  7856. }
  7857. if (typeof obj === 'object') {
  7858. return Object.keys(obj).map(function (k) {
  7859. var ks = encodeURIComponent(stringifyPrimitive(k)) + eq;
  7860. if (Array.isArray(obj[k])) {
  7861. return obj[k].map(function (v) {
  7862. return ks + encodeURIComponent(stringifyPrimitive(v));
  7863. }).join(sep);
  7864. } else {
  7865. return ks + encodeURIComponent(stringifyPrimitive(obj[k]));
  7866. }
  7867. }).filter(Boolean).join(sep);
  7868. }
  7869. if (!name) return '';
  7870. return encodeURIComponent(stringifyPrimitive(name)) + eq + encodeURIComponent(stringifyPrimitive(obj));
  7871. };
  7872. var xhrRes = function (err, xhr, body) {
  7873. var headers = {};
  7874. xhr.getAllResponseHeaders().trim().split('\n').forEach(function (item) {
  7875. if (item) {
  7876. var index = item.indexOf(':');
  7877. var key = item.substr(0, index).trim().toLowerCase();
  7878. var val = item.substr(index + 1).trim();
  7879. headers[key] = val;
  7880. }
  7881. });
  7882. return {
  7883. error: err,
  7884. statusCode: xhr.status,
  7885. statusMessage: xhr.statusText,
  7886. headers: headers,
  7887. body: body
  7888. };
  7889. };
  7890. var xhrBody = function (xhr, dataType) {
  7891. return !dataType && dataType === 'text' ? xhr.responseText : xhr.response;
  7892. };
  7893. var request = function (opt, callback) {
  7894. // method
  7895. var method = (opt.method || 'GET').toUpperCase();
  7896. // url、qs
  7897. var url = opt.url;
  7898. if (opt.qs) {
  7899. var qsStr = queryStringify(opt.qs);
  7900. if (qsStr) {
  7901. url += (url.indexOf('?') === -1 ? '?' : '&') + qsStr;
  7902. }
  7903. }
  7904. // 创建 ajax 实例
  7905. var xhr = new XMLHttpRequest();
  7906. xhr.open(method, url, true);
  7907. xhr.responseType = opt.dataType || 'text';
  7908. // 处理 headers
  7909. var headers = opt.headers;
  7910. if (headers) {
  7911. for (var key in headers) {
  7912. if (headers.hasOwnProperty(key) && key.toLowerCase() !== 'content-length' && key.toLowerCase() !== 'user-agent' && key.toLowerCase() !== 'origin' && key.toLowerCase() !== 'host') {
  7913. xhr.setRequestHeader(key, headers[key]);
  7914. }
  7915. }
  7916. }
  7917. // onprogress
  7918. if (opt.onProgress && xhr.upload) xhr.upload.onprogress = opt.onProgress;
  7919. if (opt.onDownloadProgress) xhr.onprogress = opt.onDownloadProgress;
  7920. // success 2xx/3xx/4xx
  7921. xhr.onload = function () {
  7922. callback(xhrRes(null, xhr, xhrBody(xhr, opt.dataType)));
  7923. };
  7924. // error 5xx/0 (网络错误、跨域报错、Https connect-src 限制的报错时 statusCode 为 0)
  7925. xhr.onerror = function (err) {
  7926. var body = xhrBody(xhr, opt.dataType);
  7927. if (body) {
  7928. // 5xx
  7929. callback(xhrRes(null, xhr, body));
  7930. } else {
  7931. // 0
  7932. var error = xhr.statusText;
  7933. if (!error && xhr.status === 0) error = new Error('CORS blocked or network error');
  7934. callback(xhrRes(error, xhr, body));
  7935. }
  7936. };
  7937. // send
  7938. xhr.send(opt.body || '');
  7939. // 返回 ajax 实例,用于外部调用 xhr.abort
  7940. return xhr;
  7941. };
  7942. module.exports = request;
  7943. /***/ }),
  7944. /* 18 */
  7945. /***/ (function(module, exports, __webpack_require__) {
  7946. var session = __webpack_require__(4);
  7947. var Async = __webpack_require__(19);
  7948. var EventProxy = __webpack_require__(3).EventProxy;
  7949. var util = __webpack_require__(0);
  7950. // 文件分块上传全过程,暴露的分块上传接口
  7951. function sliceUploadFile(params, callback) {
  7952. var self = this;
  7953. var ep = new EventProxy();
  7954. var TaskId = params.TaskId;
  7955. var Bucket = params.Bucket;
  7956. var Region = params.Region;
  7957. var Key = params.Key;
  7958. var Body = params.Body;
  7959. var ChunkSize = params.ChunkSize || params.SliceSize || self.options.ChunkSize;
  7960. var AsyncLimit = params.AsyncLimit;
  7961. var StorageClass = params.StorageClass;
  7962. var ServerSideEncryption = params.ServerSideEncryption;
  7963. var FileSize;
  7964. var onProgress;
  7965. var onHashProgress = params.onHashProgress;
  7966. // 上传过程中出现错误,返回错误
  7967. ep.on('error', function (err) {
  7968. if (!self._isRunningTask(TaskId)) return;
  7969. var _err = util.extend({
  7970. UploadId: params.UploadData.UploadId || ''
  7971. }, err);
  7972. return callback(_err);
  7973. });
  7974. // 上传分块完成,开始 uploadSliceComplete 操作
  7975. ep.on('upload_complete', function (UploadCompleteData) {
  7976. var _UploadCompleteData = util.extend({
  7977. UploadId: params.UploadData.UploadId || ''
  7978. }, UploadCompleteData);
  7979. callback(null, _UploadCompleteData);
  7980. });
  7981. // 上传分块完成,开始 uploadSliceComplete 操作
  7982. ep.on('upload_slice_complete', function (UploadData) {
  7983. var metaHeaders = {};
  7984. util.each(params.Headers, function (val, k) {
  7985. var shortKey = k.toLowerCase();
  7986. if (shortKey.indexOf('x-cos-meta-') === 0 || shortKey === 'pic-operations') metaHeaders[k] = val;
  7987. });
  7988. uploadSliceComplete.call(self, {
  7989. Bucket: Bucket,
  7990. Region: Region,
  7991. Key: Key,
  7992. UploadId: UploadData.UploadId,
  7993. SliceList: UploadData.SliceList,
  7994. Headers: metaHeaders
  7995. }, function (err, data) {
  7996. if (!self._isRunningTask(TaskId)) return;
  7997. session.removeUsing(UploadData.UploadId);
  7998. if (err) {
  7999. onProgress(null, true);
  8000. return ep.emit('error', err);
  8001. }
  8002. session.removeUploadId.call(self, UploadData.UploadId);
  8003. onProgress({ loaded: FileSize, total: FileSize }, true);
  8004. ep.emit('upload_complete', data);
  8005. });
  8006. });
  8007. // 获取 UploadId 完成,开始上传每个分片
  8008. ep.on('get_upload_data_finish', function (UploadData) {
  8009. // 处理 UploadId 缓存
  8010. var uuid = session.getFileId(Body, params.ChunkSize, Bucket, Key);
  8011. uuid && session.saveUploadId.call(self, uuid, UploadData.UploadId, self.options.UploadIdCacheLimit); // 缓存 UploadId
  8012. session.setUsing(UploadData.UploadId); // 标记 UploadId 为正在使用
  8013. // 获取 UploadId
  8014. onProgress(null, true); // 任务状态开始 uploading
  8015. uploadSliceList.call(self, {
  8016. TaskId: TaskId,
  8017. Bucket: Bucket,
  8018. Region: Region,
  8019. Key: Key,
  8020. Body: Body,
  8021. FileSize: FileSize,
  8022. SliceSize: ChunkSize,
  8023. AsyncLimit: AsyncLimit,
  8024. ServerSideEncryption: ServerSideEncryption,
  8025. UploadData: UploadData,
  8026. onProgress: onProgress
  8027. }, function (err, data) {
  8028. if (!self._isRunningTask(TaskId)) return;
  8029. if (err) {
  8030. onProgress(null, true);
  8031. return ep.emit('error', err);
  8032. }
  8033. ep.emit('upload_slice_complete', data);
  8034. });
  8035. });
  8036. // 开始获取文件 UploadId,里面会视情况计算 ETag,并比对,保证文件一致性,也优化上传
  8037. ep.on('get_file_size_finish', function () {
  8038. onProgress = util.throttleOnProgress.call(self, FileSize, params.onProgress);
  8039. if (params.UploadData.UploadId) {
  8040. ep.emit('get_upload_data_finish', params.UploadData);
  8041. } else {
  8042. var _params = util.extend({
  8043. TaskId: TaskId,
  8044. Bucket: Bucket,
  8045. Region: Region,
  8046. Key: Key,
  8047. Headers: params.Headers,
  8048. StorageClass: StorageClass,
  8049. Body: Body,
  8050. FileSize: FileSize,
  8051. SliceSize: ChunkSize,
  8052. onHashProgress: onHashProgress
  8053. }, params);
  8054. getUploadIdAndPartList.call(self, _params, function (err, UploadData) {
  8055. if (!self._isRunningTask(TaskId)) return;
  8056. if (err) return ep.emit('error', err);
  8057. params.UploadData.UploadId = UploadData.UploadId;
  8058. params.UploadData.PartList = UploadData.PartList;
  8059. ep.emit('get_upload_data_finish', params.UploadData);
  8060. });
  8061. }
  8062. });
  8063. // 获取上传文件大小
  8064. FileSize = params.ContentLength;
  8065. delete params.ContentLength;
  8066. !params.Headers && (params.Headers = {});
  8067. util.each(params.Headers, function (item, key) {
  8068. if (key.toLowerCase() === 'content-length') {
  8069. delete params.Headers[key];
  8070. }
  8071. });
  8072. // 控制分片大小
  8073. (function () {
  8074. var SIZE = [1, 2, 4, 8, 16, 32, 64, 128, 256, 512, 1024, 1024 * 2, 1024 * 4, 1024 * 5];
  8075. var AutoChunkSize = 1024 * 1024;
  8076. for (var i = 0; i < SIZE.length; i++) {
  8077. AutoChunkSize = SIZE[i] * 1024 * 1024;
  8078. if (FileSize / AutoChunkSize <= self.options.MaxPartNumber) break;
  8079. }
  8080. params.ChunkSize = params.SliceSize = ChunkSize = Math.max(ChunkSize, AutoChunkSize);
  8081. })();
  8082. // 开始上传
  8083. if (FileSize === 0) {
  8084. params.Body = '';
  8085. params.ContentLength = 0;
  8086. params.SkipTask = true;
  8087. self.putObject(params, callback);
  8088. } else {
  8089. ep.emit('get_file_size_finish');
  8090. }
  8091. }
  8092. // 获取上传任务的 UploadId
  8093. function getUploadIdAndPartList(params, callback) {
  8094. var TaskId = params.TaskId;
  8095. var Bucket = params.Bucket;
  8096. var Region = params.Region;
  8097. var Key = params.Key;
  8098. var StorageClass = params.StorageClass;
  8099. var self = this;
  8100. // 计算 ETag
  8101. var ETagMap = {};
  8102. var FileSize = params.FileSize;
  8103. var SliceSize = params.SliceSize;
  8104. var SliceCount = Math.ceil(FileSize / SliceSize);
  8105. var FinishSliceCount = 0;
  8106. var FinishSize = 0;
  8107. var onHashProgress = util.throttleOnProgress.call(self, FileSize, params.onHashProgress);
  8108. var getChunkETag = function (PartNumber, callback) {
  8109. var start = SliceSize * (PartNumber - 1);
  8110. var end = Math.min(start + SliceSize, FileSize);
  8111. var ChunkSize = end - start;
  8112. if (ETagMap[PartNumber]) {
  8113. callback(null, {
  8114. PartNumber: PartNumber,
  8115. ETag: ETagMap[PartNumber],
  8116. Size: ChunkSize
  8117. });
  8118. } else {
  8119. util.fileSlice(params.Body, start, end, false, function (chunkItem) {
  8120. util.getFileMd5(chunkItem, function (err, md5) {
  8121. if (err) return callback(util.error(err));
  8122. var ETag = '"' + md5 + '"';
  8123. ETagMap[PartNumber] = ETag;
  8124. FinishSliceCount += 1;
  8125. FinishSize += ChunkSize;
  8126. onHashProgress({ loaded: FinishSize, total: FileSize });
  8127. callback(null, {
  8128. PartNumber: PartNumber,
  8129. ETag: ETag,
  8130. Size: ChunkSize
  8131. });
  8132. });
  8133. });
  8134. }
  8135. };
  8136. // 通过和文件的 md5 对比,判断 UploadId 是否可用
  8137. var isAvailableUploadList = function (PartList, callback) {
  8138. var PartCount = PartList.length;
  8139. // 如果没有分片,通过
  8140. if (PartCount === 0) {
  8141. return callback(null, true);
  8142. }
  8143. // 检查分片数量
  8144. if (PartCount > SliceCount) {
  8145. return callback(null, false);
  8146. }
  8147. // 检查分片大小
  8148. if (PartCount > 1) {
  8149. var PartSliceSize = Math.max(PartList[0].Size, PartList[1].Size);
  8150. if (PartSliceSize !== SliceSize) {
  8151. return callback(null, false);
  8152. }
  8153. }
  8154. // 逐个分片计算并检查 ETag 是否一致
  8155. var next = function (index) {
  8156. if (index < PartCount) {
  8157. var Part = PartList[index];
  8158. getChunkETag(Part.PartNumber, function (err, chunk) {
  8159. if (chunk && chunk.ETag === Part.ETag && chunk.Size === Part.Size) {
  8160. next(index + 1);
  8161. } else {
  8162. callback(null, false);
  8163. }
  8164. });
  8165. } else {
  8166. callback(null, true);
  8167. }
  8168. };
  8169. next(0);
  8170. };
  8171. var ep = new EventProxy();
  8172. ep.on('error', function (errData) {
  8173. if (!self._isRunningTask(TaskId)) return;
  8174. return callback(errData);
  8175. });
  8176. // 存在 UploadId
  8177. ep.on('upload_id_available', function (UploadData) {
  8178. // 转换成 map
  8179. var map = {};
  8180. var list = [];
  8181. util.each(UploadData.PartList, function (item) {
  8182. map[item.PartNumber] = item;
  8183. });
  8184. for (var PartNumber = 1; PartNumber <= SliceCount; PartNumber++) {
  8185. var item = map[PartNumber];
  8186. if (item) {
  8187. item.PartNumber = PartNumber;
  8188. item.Uploaded = true;
  8189. } else {
  8190. item = {
  8191. PartNumber: PartNumber,
  8192. ETag: null,
  8193. Uploaded: false
  8194. };
  8195. }
  8196. list.push(item);
  8197. }
  8198. UploadData.PartList = list;
  8199. callback(null, UploadData);
  8200. });
  8201. // 不存在 UploadId, 初始化生成 UploadId
  8202. ep.on('no_available_upload_id', function () {
  8203. if (!self._isRunningTask(TaskId)) return;
  8204. var _params = util.extend({
  8205. Bucket: Bucket,
  8206. Region: Region,
  8207. Key: Key,
  8208. Headers: util.clone(params.Headers),
  8209. Query: util.clone(params.Query),
  8210. StorageClass: StorageClass,
  8211. Body: params.Body
  8212. }, params);
  8213. self.multipartInit(_params, function (err, data) {
  8214. if (!self._isRunningTask(TaskId)) return;
  8215. if (err) return ep.emit('error', err);
  8216. var UploadId = data.UploadId;
  8217. if (!UploadId) {
  8218. return callback(util.error(new Error('no such upload id')));
  8219. }
  8220. ep.emit('upload_id_available', { UploadId: UploadId, PartList: [] });
  8221. });
  8222. });
  8223. // 如果已存在 UploadId,找一个可以用的 UploadId
  8224. ep.on('has_and_check_upload_id', function (UploadIdList) {
  8225. // 串行地,找一个内容一致的 UploadId
  8226. UploadIdList = UploadIdList.reverse();
  8227. Async.eachLimit(UploadIdList, 1, function (UploadId, asyncCallback) {
  8228. if (!self._isRunningTask(TaskId)) return;
  8229. // 如果正在上传,跳过
  8230. if (session.using[UploadId]) {
  8231. asyncCallback(); // 检查下一个 UploadId
  8232. return;
  8233. }
  8234. // 判断 UploadId 是否可用
  8235. wholeMultipartListPart.call(self, {
  8236. Bucket: Bucket,
  8237. Region: Region,
  8238. Key: Key,
  8239. UploadId: UploadId
  8240. }, function (err, PartListData) {
  8241. if (!self._isRunningTask(TaskId)) return;
  8242. if (err) {
  8243. session.removeUsing(UploadId);
  8244. return ep.emit('error', err);
  8245. }
  8246. var PartList = PartListData.PartList;
  8247. PartList.forEach(function (item) {
  8248. item.PartNumber *= 1;
  8249. item.Size *= 1;
  8250. item.ETag = item.ETag || '';
  8251. });
  8252. isAvailableUploadList(PartList, function (err, isAvailable) {
  8253. if (!self._isRunningTask(TaskId)) return;
  8254. if (err) return ep.emit('error', err);
  8255. if (isAvailable) {
  8256. asyncCallback({
  8257. UploadId: UploadId,
  8258. PartList: PartList
  8259. }); // 马上结束
  8260. } else {
  8261. asyncCallback(); // 检查下一个 UploadId
  8262. }
  8263. });
  8264. });
  8265. }, function (AvailableUploadData) {
  8266. if (!self._isRunningTask(TaskId)) return;
  8267. onHashProgress(null, true);
  8268. if (AvailableUploadData && AvailableUploadData.UploadId) {
  8269. ep.emit('upload_id_available', AvailableUploadData);
  8270. } else {
  8271. ep.emit('no_available_upload_id');
  8272. }
  8273. });
  8274. });
  8275. // 在本地缓存找可用的 UploadId
  8276. ep.on('seek_local_avail_upload_id', function (RemoteUploadIdList) {
  8277. // 在本地找可用的 UploadId
  8278. var uuid = session.getFileId(params.Body, params.ChunkSize, Bucket, Key);
  8279. var LocalUploadIdList = session.getUploadIdList.call(self, uuid);
  8280. if (!uuid || !LocalUploadIdList) {
  8281. ep.emit('has_and_check_upload_id', RemoteUploadIdList);
  8282. return;
  8283. }
  8284. var next = function (index) {
  8285. // 如果本地找不到可用 UploadId,再一个个遍历校验远端
  8286. if (index >= LocalUploadIdList.length) {
  8287. ep.emit('has_and_check_upload_id', RemoteUploadIdList);
  8288. return;
  8289. }
  8290. var UploadId = LocalUploadIdList[index];
  8291. // 如果不在远端 UploadId 列表里,跳过并删除
  8292. if (!util.isInArray(RemoteUploadIdList, UploadId)) {
  8293. session.removeUploadId.call(self, UploadId);
  8294. next(index + 1);
  8295. return;
  8296. }
  8297. // 如果正在上传,跳过
  8298. if (session.using[UploadId]) {
  8299. next(index + 1);
  8300. return;
  8301. }
  8302. // 判断 UploadId 是否存在线上
  8303. wholeMultipartListPart.call(self, {
  8304. Bucket: Bucket,
  8305. Region: Region,
  8306. Key: Key,
  8307. UploadId: UploadId
  8308. }, function (err, PartListData) {
  8309. if (!self._isRunningTask(TaskId)) return;
  8310. if (err) {
  8311. // 如果 UploadId 获取会出错,跳过并删除
  8312. session.removeUploadId.call(self, UploadId);
  8313. next(index + 1);
  8314. } else {
  8315. // 找到可用 UploadId
  8316. ep.emit('upload_id_available', {
  8317. UploadId: UploadId,
  8318. PartList: PartListData.PartList
  8319. });
  8320. }
  8321. });
  8322. };
  8323. next(0);
  8324. });
  8325. // 获取线上 UploadId 列表
  8326. ep.on('get_remote_upload_id_list', function () {
  8327. // 获取符合条件的 UploadId 列表,因为同一个文件可以有多个上传任务。
  8328. wholeMultipartList.call(self, {
  8329. Bucket: Bucket,
  8330. Region: Region,
  8331. Key: Key
  8332. }, function (err, data) {
  8333. if (!self._isRunningTask(TaskId)) return;
  8334. if (err) return ep.emit('error', err);
  8335. // 整理远端 UploadId 列表
  8336. var RemoteUploadIdList = util.filter(data.UploadList, function (item) {
  8337. return item.Key === Key && (!StorageClass || item.StorageClass.toUpperCase() === StorageClass.toUpperCase());
  8338. }).reverse().map(function (item) {
  8339. return item.UploadId || item.UploadID;
  8340. });
  8341. if (RemoteUploadIdList.length) {
  8342. ep.emit('seek_local_avail_upload_id', RemoteUploadIdList);
  8343. } else {
  8344. // 远端没有 UploadId,清理缓存的 UploadId
  8345. var uuid = session.getFileId(params.Body, params.ChunkSize, Bucket, Key),
  8346. LocalUploadIdList;
  8347. if (uuid && (LocalUploadIdList = session.getUploadIdList.call(self, uuid))) {
  8348. util.each(LocalUploadIdList, function (UploadId) {
  8349. session.removeUploadId.call(self, UploadId);
  8350. });
  8351. }
  8352. ep.emit('no_available_upload_id');
  8353. }
  8354. });
  8355. });
  8356. // 开始找可用 UploadId
  8357. ep.emit('get_remote_upload_id_list');
  8358. }
  8359. // 获取符合条件的全部上传任务 (条件包括 Bucket, Region, Prefix)
  8360. function wholeMultipartList(params, callback) {
  8361. var self = this;
  8362. var UploadList = [];
  8363. var sendParams = {
  8364. Bucket: params.Bucket,
  8365. Region: params.Region,
  8366. Prefix: params.Key
  8367. };
  8368. var next = function () {
  8369. self.multipartList(sendParams, function (err, data) {
  8370. if (err) return callback(err);
  8371. UploadList.push.apply(UploadList, data.Upload || []);
  8372. if (data.IsTruncated === 'true') {
  8373. // 列表不完整
  8374. sendParams.KeyMarker = data.NextKeyMarker;
  8375. sendParams.UploadIdMarker = data.NextUploadIdMarker;
  8376. next();
  8377. } else {
  8378. callback(null, { UploadList: UploadList });
  8379. }
  8380. });
  8381. };
  8382. next();
  8383. }
  8384. // 获取指定上传任务的分块列表
  8385. function wholeMultipartListPart(params, callback) {
  8386. var self = this;
  8387. var PartList = [];
  8388. var sendParams = {
  8389. Bucket: params.Bucket,
  8390. Region: params.Region,
  8391. Key: params.Key,
  8392. UploadId: params.UploadId
  8393. };
  8394. var next = function () {
  8395. self.multipartListPart(sendParams, function (err, data) {
  8396. if (err) return callback(err);
  8397. PartList.push.apply(PartList, data.Part || []);
  8398. if (data.IsTruncated === 'true') {
  8399. // 列表不完整
  8400. sendParams.PartNumberMarker = data.NextPartNumberMarker;
  8401. next();
  8402. } else {
  8403. callback(null, { PartList: PartList });
  8404. }
  8405. });
  8406. };
  8407. next();
  8408. }
  8409. // 上传文件分块,包括
  8410. /*
  8411. UploadId (上传任务编号)
  8412. AsyncLimit (并发量),
  8413. SliceList (上传的分块数组),
  8414. FilePath (本地文件的位置),
  8415. SliceSize (文件分块大小)
  8416. FileSize (文件大小)
  8417. onProgress (上传成功之后的回调函数)
  8418. */
  8419. function uploadSliceList(params, cb) {
  8420. var self = this;
  8421. var TaskId = params.TaskId;
  8422. var Bucket = params.Bucket;
  8423. var Region = params.Region;
  8424. var Key = params.Key;
  8425. var UploadData = params.UploadData;
  8426. var FileSize = params.FileSize;
  8427. var SliceSize = params.SliceSize;
  8428. var ChunkParallel = Math.min(params.AsyncLimit || self.options.ChunkParallelLimit || 1, 256);
  8429. var Body = params.Body;
  8430. var SliceCount = Math.ceil(FileSize / SliceSize);
  8431. var FinishSize = 0;
  8432. var ServerSideEncryption = params.ServerSideEncryption;
  8433. var needUploadSlices = util.filter(UploadData.PartList, function (SliceItem) {
  8434. if (SliceItem['Uploaded']) {
  8435. FinishSize += SliceItem['PartNumber'] >= SliceCount ? FileSize % SliceSize || SliceSize : SliceSize;
  8436. }
  8437. return !SliceItem['Uploaded'];
  8438. });
  8439. var onProgress = params.onProgress;
  8440. Async.eachLimit(needUploadSlices, ChunkParallel, function (SliceItem, asyncCallback) {
  8441. if (!self._isRunningTask(TaskId)) return;
  8442. var PartNumber = SliceItem['PartNumber'];
  8443. var currentSize = Math.min(FileSize, SliceItem['PartNumber'] * SliceSize) - (SliceItem['PartNumber'] - 1) * SliceSize;
  8444. var preAddSize = 0;
  8445. uploadSliceItem.call(self, {
  8446. TaskId: TaskId,
  8447. Bucket: Bucket,
  8448. Region: Region,
  8449. Key: Key,
  8450. SliceSize: SliceSize,
  8451. FileSize: FileSize,
  8452. PartNumber: PartNumber,
  8453. ServerSideEncryption: ServerSideEncryption,
  8454. Body: Body,
  8455. UploadData: UploadData,
  8456. onProgress: function (data) {
  8457. FinishSize += data.loaded - preAddSize;
  8458. preAddSize = data.loaded;
  8459. onProgress({ loaded: FinishSize, total: FileSize });
  8460. }
  8461. }, function (err, data) {
  8462. if (!self._isRunningTask(TaskId)) return;
  8463. if (!err && !data.ETag) err = 'get ETag error, please add "ETag" to CORS ExposeHeader setting.( 获取ETag失败,请在CORS ExposeHeader设置中添加ETag,请参考文档:https://cloud.tencent.com/document/product/436/13318 )';
  8464. if (err) {
  8465. FinishSize -= preAddSize;
  8466. } else {
  8467. FinishSize += currentSize - preAddSize;
  8468. SliceItem.ETag = data.ETag;
  8469. }
  8470. onProgress({ loaded: FinishSize, total: FileSize });
  8471. asyncCallback(err || null, data);
  8472. });
  8473. }, function (err) {
  8474. if (!self._isRunningTask(TaskId)) return;
  8475. if (err) return cb(err);
  8476. cb(null, {
  8477. UploadId: UploadData.UploadId,
  8478. SliceList: UploadData.PartList
  8479. });
  8480. });
  8481. }
  8482. // 上传指定分片
  8483. function uploadSliceItem(params, callback) {
  8484. var self = this;
  8485. var TaskId = params.TaskId;
  8486. var Bucket = params.Bucket;
  8487. var Region = params.Region;
  8488. var Key = params.Key;
  8489. var FileSize = params.FileSize;
  8490. var FileBody = params.Body;
  8491. var PartNumber = params.PartNumber * 1;
  8492. var SliceSize = params.SliceSize;
  8493. var ServerSideEncryption = params.ServerSideEncryption;
  8494. var UploadData = params.UploadData;
  8495. var ChunkRetryTimes = self.options.ChunkRetryTimes + 1;
  8496. var start = SliceSize * (PartNumber - 1);
  8497. var ContentLength = SliceSize;
  8498. var end = start + SliceSize;
  8499. if (end > FileSize) {
  8500. end = FileSize;
  8501. ContentLength = end - start;
  8502. }
  8503. var PartItem = UploadData.PartList[PartNumber - 1];
  8504. Async.retry(ChunkRetryTimes, function (tryCallback) {
  8505. if (!self._isRunningTask(TaskId)) return;
  8506. util.fileSlice(FileBody, start, end, true, function (Body) {
  8507. self.multipartUpload({
  8508. TaskId: TaskId,
  8509. Bucket: Bucket,
  8510. Region: Region,
  8511. Key: Key,
  8512. ContentLength: ContentLength,
  8513. PartNumber: PartNumber,
  8514. UploadId: UploadData.UploadId,
  8515. ServerSideEncryption: ServerSideEncryption,
  8516. Body: Body,
  8517. onProgress: params.onProgress
  8518. }, function (err, data) {
  8519. if (!self._isRunningTask(TaskId)) return;
  8520. if (err) return tryCallback(err);
  8521. PartItem.Uploaded = true;
  8522. return tryCallback(null, data);
  8523. });
  8524. });
  8525. }, function (err, data) {
  8526. if (!self._isRunningTask(TaskId)) return;
  8527. return callback(err, data);
  8528. });
  8529. }
  8530. // 完成分块上传
  8531. function uploadSliceComplete(params, callback) {
  8532. var Bucket = params.Bucket;
  8533. var Region = params.Region;
  8534. var Key = params.Key;
  8535. var UploadId = params.UploadId;
  8536. var SliceList = params.SliceList;
  8537. var self = this;
  8538. var ChunkRetryTimes = this.options.ChunkRetryTimes + 1;
  8539. var Headers = params.Headers;
  8540. var Parts = SliceList.map(function (item) {
  8541. return {
  8542. PartNumber: item.PartNumber,
  8543. ETag: item.ETag
  8544. };
  8545. });
  8546. // 完成上传的请求也做重试
  8547. Async.retry(ChunkRetryTimes, function (tryCallback) {
  8548. self.multipartComplete({
  8549. Bucket: Bucket,
  8550. Region: Region,
  8551. Key: Key,
  8552. UploadId: UploadId,
  8553. Parts: Parts,
  8554. Headers: Headers
  8555. }, tryCallback);
  8556. }, function (err, data) {
  8557. callback(err, data);
  8558. });
  8559. }
  8560. // 抛弃分块上传任务
  8561. /*
  8562. AsyncLimit (抛弃上传任务的并发量),
  8563. UploadId (上传任务的编号,当 Level 为 task 时候需要)
  8564. Level (抛弃分块上传任务的级别,task : 抛弃指定的上传任务,file : 抛弃指定的文件对应的上传任务,其他值 :抛弃指定Bucket 的全部上传任务)
  8565. */
  8566. function abortUploadTask(params, callback) {
  8567. var Bucket = params.Bucket;
  8568. var Region = params.Region;
  8569. var Key = params.Key;
  8570. var UploadId = params.UploadId;
  8571. var Level = params.Level || 'task';
  8572. var AsyncLimit = params.AsyncLimit;
  8573. var self = this;
  8574. var ep = new EventProxy();
  8575. ep.on('error', function (errData) {
  8576. return callback(errData);
  8577. });
  8578. // 已经获取到需要抛弃的任务列表
  8579. ep.on('get_abort_array', function (AbortArray) {
  8580. abortUploadTaskArray.call(self, {
  8581. Bucket: Bucket,
  8582. Region: Region,
  8583. Key: Key,
  8584. Headers: params.Headers,
  8585. AsyncLimit: AsyncLimit,
  8586. AbortArray: AbortArray
  8587. }, callback);
  8588. });
  8589. if (Level === 'bucket') {
  8590. // Bucket 级别的任务抛弃,抛弃该 Bucket 下的全部上传任务
  8591. wholeMultipartList.call(self, {
  8592. Bucket: Bucket,
  8593. Region: Region
  8594. }, function (err, data) {
  8595. if (err) return callback(err);
  8596. ep.emit('get_abort_array', data.UploadList || []);
  8597. });
  8598. } else if (Level === 'file') {
  8599. // 文件级别的任务抛弃,抛弃该文件的全部上传任务
  8600. if (!Key) return callback(util.error(new Error('abort_upload_task_no_key')));
  8601. wholeMultipartList.call(self, {
  8602. Bucket: Bucket,
  8603. Region: Region,
  8604. Key: Key
  8605. }, function (err, data) {
  8606. if (err) return callback(err);
  8607. ep.emit('get_abort_array', data.UploadList || []);
  8608. });
  8609. } else if (Level === 'task') {
  8610. // 单个任务级别的任务抛弃,抛弃指定 UploadId 的上传任务
  8611. if (!UploadId) return callback(util.error(new Error('abort_upload_task_no_id')));
  8612. if (!Key) return callback(util.error(new Error('abort_upload_task_no_key')));
  8613. ep.emit('get_abort_array', [{
  8614. Key: Key,
  8615. UploadId: UploadId
  8616. }]);
  8617. } else {
  8618. return callback(util.error(new Error('abort_unknown_level')));
  8619. }
  8620. }
  8621. // 批量抛弃分块上传任务
  8622. function abortUploadTaskArray(params, callback) {
  8623. var Bucket = params.Bucket;
  8624. var Region = params.Region;
  8625. var Key = params.Key;
  8626. var AbortArray = params.AbortArray;
  8627. var AsyncLimit = params.AsyncLimit || 1;
  8628. var self = this;
  8629. var index = 0;
  8630. var resultList = new Array(AbortArray.length);
  8631. Async.eachLimit(AbortArray, AsyncLimit, function (AbortItem, nextItem) {
  8632. var eachIndex = index;
  8633. if (Key && Key !== AbortItem.Key) {
  8634. resultList[eachIndex] = { error: { KeyNotMatch: true } };
  8635. nextItem(null);
  8636. return;
  8637. }
  8638. var UploadId = AbortItem.UploadId || AbortItem.UploadID;
  8639. self.multipartAbort({
  8640. Bucket: Bucket,
  8641. Region: Region,
  8642. Key: AbortItem.Key,
  8643. Headers: params.Headers,
  8644. UploadId: UploadId
  8645. }, function (err) {
  8646. var task = {
  8647. Bucket: Bucket,
  8648. Region: Region,
  8649. Key: AbortItem.Key,
  8650. UploadId: UploadId
  8651. };
  8652. resultList[eachIndex] = { error: err, task: task };
  8653. nextItem(null);
  8654. });
  8655. index++;
  8656. }, function (err) {
  8657. if (err) return callback(err);
  8658. var successList = [];
  8659. var errorList = [];
  8660. for (var i = 0, len = resultList.length; i < len; i++) {
  8661. var item = resultList[i];
  8662. if (item['task']) {
  8663. if (item['error']) {
  8664. errorList.push(item['task']);
  8665. } else {
  8666. successList.push(item['task']);
  8667. }
  8668. }
  8669. }
  8670. return callback(null, {
  8671. successList: successList,
  8672. errorList: errorList
  8673. });
  8674. });
  8675. }
  8676. // 高级上传
  8677. function uploadFile(params, callback) {
  8678. var self = this;
  8679. // 判断多大的文件使用分片上传
  8680. var SliceSize = params.SliceSize === undefined ? self.options.SliceSize : params.SliceSize;
  8681. var taskList = [];
  8682. var Body = params.Body;
  8683. var FileSize = Body.size || Body.length || 0;
  8684. var fileInfo = { TaskId: '' };
  8685. // 整理 option,用于返回给回调
  8686. util.each(params, function (v, k) {
  8687. if (typeof v !== 'object' && typeof v !== 'function') {
  8688. fileInfo[k] = v;
  8689. }
  8690. });
  8691. // 处理文件 TaskReady
  8692. var _onTaskReady = params.onTaskReady;
  8693. var onTaskReady = function (tid) {
  8694. fileInfo.TaskId = tid;
  8695. _onTaskReady && _onTaskReady(tid);
  8696. };
  8697. params.onTaskReady = onTaskReady;
  8698. // 处理文件完成
  8699. var _onFileFinish = params.onFileFinish;
  8700. var onFileFinish = function (err, data) {
  8701. _onFileFinish && _onFileFinish(err, data, fileInfo);
  8702. callback && callback(err, data);
  8703. };
  8704. // 添加上传任务,超过阈值使用分块上传,小于等于则简单上传
  8705. var api = FileSize > SliceSize ? 'sliceUploadFile' : 'putObject';
  8706. taskList.push({
  8707. api: api,
  8708. params: params,
  8709. callback: onFileFinish
  8710. });
  8711. self._addTasks(taskList);
  8712. }
  8713. // 批量上传文件
  8714. function uploadFiles(params, callback) {
  8715. var self = this;
  8716. // 判断多大的文件使用分片上传
  8717. var SliceSize = params.SliceSize === undefined ? self.options.SliceSize : params.SliceSize;
  8718. // 汇总返回进度
  8719. var TotalSize = 0;
  8720. var TotalFinish = 0;
  8721. var onTotalProgress = util.throttleOnProgress.call(self, TotalFinish, params.onProgress);
  8722. // 汇总返回回调
  8723. var unFinishCount = params.files.length;
  8724. var _onTotalFileFinish = params.onFileFinish;
  8725. var resultList = Array(unFinishCount);
  8726. var onTotalFileFinish = function (err, data, options) {
  8727. onTotalProgress(null, true);
  8728. _onTotalFileFinish && _onTotalFileFinish(err, data, options);
  8729. resultList[options.Index] = {
  8730. options: options,
  8731. error: err,
  8732. data: data
  8733. };
  8734. if (--unFinishCount <= 0 && callback) {
  8735. callback(null, { files: resultList });
  8736. }
  8737. };
  8738. // 开始处理每个文件
  8739. var taskList = [];
  8740. util.each(params.files, function (fileParams, index) {
  8741. (function () {
  8742. // 对齐 nodejs 缩进
  8743. var Body = fileParams.Body;
  8744. var FileSize = Body.size || Body.length || 0;
  8745. var fileInfo = { Index: index, TaskId: '' };
  8746. // 更新文件总大小
  8747. TotalSize += FileSize;
  8748. // 整理 option,用于返回给回调
  8749. util.each(fileParams, function (v, k) {
  8750. if (typeof v !== 'object' && typeof v !== 'function') {
  8751. fileInfo[k] = v;
  8752. }
  8753. });
  8754. // 处理单个文件 TaskReady
  8755. var _onTaskReady = fileParams.onTaskReady;
  8756. var onTaskReady = function (tid) {
  8757. fileInfo.TaskId = tid;
  8758. _onTaskReady && _onTaskReady(tid);
  8759. };
  8760. fileParams.onTaskReady = onTaskReady;
  8761. // 处理单个文件进度
  8762. var PreAddSize = 0;
  8763. var _onProgress = fileParams.onProgress;
  8764. var onProgress = function (info) {
  8765. TotalFinish = TotalFinish - PreAddSize + info.loaded;
  8766. PreAddSize = info.loaded;
  8767. _onProgress && _onProgress(info);
  8768. onTotalProgress({ loaded: TotalFinish, total: TotalSize });
  8769. };
  8770. fileParams.onProgress = onProgress;
  8771. // 处理单个文件完成
  8772. var _onFileFinish = fileParams.onFileFinish;
  8773. var onFileFinish = function (err, data) {
  8774. _onFileFinish && _onFileFinish(err, data);
  8775. onTotalFileFinish && onTotalFileFinish(err, data, fileInfo);
  8776. };
  8777. // 添加上传任务
  8778. var api = FileSize > SliceSize ? 'sliceUploadFile' : 'putObject';
  8779. taskList.push({
  8780. api: api,
  8781. params: fileParams,
  8782. callback: onFileFinish
  8783. });
  8784. })();
  8785. });
  8786. self._addTasks(taskList);
  8787. }
  8788. // 分片复制文件
  8789. function sliceCopyFile(params, callback) {
  8790. var ep = new EventProxy();
  8791. var self = this;
  8792. var Bucket = params.Bucket;
  8793. var Region = params.Region;
  8794. var Key = params.Key;
  8795. var CopySource = params.CopySource;
  8796. var m = CopySource.match(/^([^.]+-\d+)\.cos(v6)?\.([^.]+)\.[^/]+\/(.+)$/);
  8797. if (!m) {
  8798. callback(util.error(new Error('CopySource format error')));
  8799. return;
  8800. }
  8801. var SourceBucket = m[1];
  8802. var SourceRegion = m[3];
  8803. var SourceKey = decodeURIComponent(m[4]);
  8804. var CopySliceSize = params.CopySliceSize === undefined ? self.options.CopySliceSize : params.CopySliceSize;
  8805. CopySliceSize = Math.max(0, CopySliceSize);
  8806. var ChunkSize = params.CopyChunkSize || this.options.CopyChunkSize;
  8807. var ChunkParallel = this.options.CopyChunkParallelLimit;
  8808. var FinishSize = 0;
  8809. var FileSize;
  8810. var onProgress;
  8811. // 分片复制完成,开始 multipartComplete 操作
  8812. ep.on('copy_slice_complete', function (UploadData) {
  8813. var metaHeaders = {};
  8814. util.each(params.Headers, function (val, k) {
  8815. if (k.toLowerCase().indexOf('x-cos-meta-') === 0) metaHeaders[k] = val;
  8816. });
  8817. var Parts = util.map(UploadData.PartList, function (item) {
  8818. return {
  8819. PartNumber: item.PartNumber,
  8820. ETag: item.ETag
  8821. };
  8822. });
  8823. self.multipartComplete({
  8824. Bucket: Bucket,
  8825. Region: Region,
  8826. Key: Key,
  8827. UploadId: UploadData.UploadId,
  8828. Parts: Parts
  8829. }, function (err, data) {
  8830. if (err) {
  8831. onProgress(null, true);
  8832. return callback(err);
  8833. }
  8834. onProgress({ loaded: FileSize, total: FileSize }, true);
  8835. callback(null, data);
  8836. });
  8837. });
  8838. ep.on('get_copy_data_finish', function (UploadData) {
  8839. Async.eachLimit(UploadData.PartList, ChunkParallel, function (SliceItem, asyncCallback) {
  8840. var PartNumber = SliceItem.PartNumber;
  8841. var CopySourceRange = SliceItem.CopySourceRange;
  8842. var currentSize = SliceItem.end - SliceItem.start;
  8843. copySliceItem.call(self, {
  8844. Bucket: Bucket,
  8845. Region: Region,
  8846. Key: Key,
  8847. CopySource: CopySource,
  8848. UploadId: UploadData.UploadId,
  8849. PartNumber: PartNumber,
  8850. CopySourceRange: CopySourceRange
  8851. }, function (err, data) {
  8852. if (err) return asyncCallback(err);
  8853. FinishSize += currentSize;
  8854. onProgress({ loaded: FinishSize, total: FileSize });
  8855. SliceItem.ETag = data.ETag;
  8856. asyncCallback(err || null, data);
  8857. });
  8858. }, function (err) {
  8859. if (err) {
  8860. onProgress(null, true);
  8861. return callback(err);
  8862. }
  8863. ep.emit('copy_slice_complete', UploadData);
  8864. });
  8865. });
  8866. ep.on('get_file_size_finish', function (SourceHeaders) {
  8867. // 控制分片大小
  8868. (function () {
  8869. var SIZE = [1, 2, 4, 8, 16, 32, 64, 128, 256, 512, 1024, 1024 * 2, 1024 * 4, 1024 * 5];
  8870. var AutoChunkSize = 1024 * 1024;
  8871. for (var i = 0; i < SIZE.length; i++) {
  8872. AutoChunkSize = SIZE[i] * 1024 * 1024;
  8873. if (FileSize / AutoChunkSize <= self.options.MaxPartNumber) break;
  8874. }
  8875. params.ChunkSize = ChunkSize = Math.max(ChunkSize, AutoChunkSize);
  8876. var ChunkCount = Math.ceil(FileSize / ChunkSize);
  8877. var list = [];
  8878. for (var partNumber = 1; partNumber <= ChunkCount; partNumber++) {
  8879. var start = (partNumber - 1) * ChunkSize;
  8880. var end = partNumber * ChunkSize < FileSize ? partNumber * ChunkSize - 1 : FileSize - 1;
  8881. var item = {
  8882. PartNumber: partNumber,
  8883. start: start,
  8884. end: end,
  8885. CopySourceRange: "bytes=" + start + "-" + end
  8886. };
  8887. list.push(item);
  8888. }
  8889. params.PartList = list;
  8890. })();
  8891. var TargetHeader;
  8892. if (params.Headers['x-cos-metadata-directive'] === 'Replaced') {
  8893. TargetHeader = params.Headers;
  8894. } else {
  8895. TargetHeader = SourceHeaders;
  8896. }
  8897. TargetHeader['x-cos-storage-class'] = params.Headers['x-cos-storage-class'] || SourceHeaders['x-cos-storage-class'];
  8898. TargetHeader = util.clearKey(TargetHeader);
  8899. /**
  8900. * 对于归档存储的对象,如果未恢复副本,则不允许 Copy
  8901. */
  8902. if (SourceHeaders['x-cos-storage-class'] === 'ARCHIVE' || SourceHeaders['x-cos-storage-class'] === 'DEEP_ARCHIVE') {
  8903. var restoreHeader = SourceHeaders['x-cos-restore'];
  8904. if (!restoreHeader || restoreHeader === 'ongoing-request="true"') {
  8905. callback(util.error(new Error('Unrestored archive object is not allowed to be copied')));
  8906. return;
  8907. }
  8908. }
  8909. /**
  8910. * 去除一些无用的头部,规避 multipartInit 出错
  8911. * 这些头部通常是在 putObjectCopy 时才使用
  8912. */
  8913. delete TargetHeader['x-cos-copy-source'];
  8914. delete TargetHeader['x-cos-metadata-directive'];
  8915. delete TargetHeader['x-cos-copy-source-If-Modified-Since'];
  8916. delete TargetHeader['x-cos-copy-source-If-Unmodified-Since'];
  8917. delete TargetHeader['x-cos-copy-source-If-Match'];
  8918. delete TargetHeader['x-cos-copy-source-If-None-Match'];
  8919. self.multipartInit({
  8920. Bucket: Bucket,
  8921. Region: Region,
  8922. Key: Key,
  8923. Headers: TargetHeader
  8924. }, function (err, data) {
  8925. if (err) return callback(err);
  8926. params.UploadId = data.UploadId;
  8927. ep.emit('get_copy_data_finish', params);
  8928. });
  8929. });
  8930. // 获取远端复制源文件的大小
  8931. self.headObject({
  8932. Bucket: SourceBucket,
  8933. Region: SourceRegion,
  8934. Key: SourceKey
  8935. }, function (err, data) {
  8936. if (err) {
  8937. if (err.statusCode && err.statusCode === 404) {
  8938. callback(util.error(err, { ErrorStatus: SourceKey + ' Not Exist' }));
  8939. } else {
  8940. callback(err);
  8941. }
  8942. return;
  8943. }
  8944. FileSize = params.FileSize = data.headers['content-length'];
  8945. if (FileSize === undefined || !FileSize) {
  8946. callback(util.error(new Error('get Content-Length error, please add "Content-Length" to CORS ExposeHeader setting.( 获取Content-Length失败,请在CORS ExposeHeader设置中添加Content-Length,请参考文档:https://cloud.tencent.com/document/product/436/13318 )')));
  8947. return;
  8948. }
  8949. onProgress = util.throttleOnProgress.call(self, FileSize, params.onProgress);
  8950. // 开始上传
  8951. if (FileSize <= CopySliceSize) {
  8952. if (!params.Headers['x-cos-metadata-directive']) {
  8953. params.Headers['x-cos-metadata-directive'] = 'Copy';
  8954. }
  8955. self.putObjectCopy(params, function (err, data) {
  8956. if (err) {
  8957. onProgress(null, true);
  8958. return callback(err);
  8959. }
  8960. onProgress({ loaded: FileSize, total: FileSize }, true);
  8961. callback(err, data);
  8962. });
  8963. } else {
  8964. var resHeaders = data.headers;
  8965. var SourceHeaders = {
  8966. 'Cache-Control': resHeaders['cache-control'],
  8967. 'Content-Disposition': resHeaders['content-disposition'],
  8968. 'Content-Encoding': resHeaders['content-encoding'],
  8969. 'Content-Type': resHeaders['content-type'],
  8970. 'Expires': resHeaders['expires'],
  8971. 'x-cos-storage-class': resHeaders['x-cos-storage-class']
  8972. };
  8973. util.each(resHeaders, function (v, k) {
  8974. var metaPrefix = 'x-cos-meta-';
  8975. if (k.indexOf(metaPrefix) === 0 && k.length > metaPrefix.length) {
  8976. SourceHeaders[k] = v;
  8977. }
  8978. });
  8979. ep.emit('get_file_size_finish', SourceHeaders);
  8980. }
  8981. });
  8982. }
  8983. // 复制指定分片
  8984. function copySliceItem(params, callback) {
  8985. var TaskId = params.TaskId;
  8986. var Bucket = params.Bucket;
  8987. var Region = params.Region;
  8988. var Key = params.Key;
  8989. var CopySource = params.CopySource;
  8990. var UploadId = params.UploadId;
  8991. var PartNumber = params.PartNumber * 1;
  8992. var CopySourceRange = params.CopySourceRange;
  8993. var ChunkRetryTimes = this.options.ChunkRetryTimes + 1;
  8994. var self = this;
  8995. Async.retry(ChunkRetryTimes, function (tryCallback) {
  8996. self.uploadPartCopy({
  8997. TaskId: TaskId,
  8998. Bucket: Bucket,
  8999. Region: Region,
  9000. Key: Key,
  9001. CopySource: CopySource,
  9002. UploadId: UploadId,
  9003. PartNumber: PartNumber,
  9004. CopySourceRange: CopySourceRange
  9005. }, function (err, data) {
  9006. tryCallback(err || null, data);
  9007. });
  9008. }, function (err, data) {
  9009. return callback(err, data);
  9010. });
  9011. }
  9012. var API_MAP = {
  9013. sliceUploadFile: sliceUploadFile,
  9014. abortUploadTask: abortUploadTask,
  9015. uploadFile: uploadFile,
  9016. uploadFiles: uploadFiles,
  9017. sliceCopyFile: sliceCopyFile
  9018. };
  9019. module.exports.init = function (COS, task) {
  9020. task.transferToTaskMethod(API_MAP, 'sliceUploadFile');
  9021. util.each(API_MAP, function (fn, apiName) {
  9022. COS.prototype[apiName] = util.apiWrapper(apiName, fn);
  9023. });
  9024. };
  9025. /***/ }),
  9026. /* 19 */
  9027. /***/ (function(module, exports) {
  9028. var eachLimit = function (arr, limit, iterator, callback) {
  9029. callback = callback || function () {};
  9030. if (!arr.length || limit <= 0) {
  9031. return callback();
  9032. }
  9033. var completed = 0;
  9034. var started = 0;
  9035. var running = 0;
  9036. (function replenish() {
  9037. if (completed >= arr.length) {
  9038. return callback();
  9039. }
  9040. while (running < limit && started < arr.length) {
  9041. started += 1;
  9042. running += 1;
  9043. iterator(arr[started - 1], function (err) {
  9044. if (err) {
  9045. callback(err);
  9046. callback = function () {};
  9047. } else {
  9048. completed += 1;
  9049. running -= 1;
  9050. if (completed >= arr.length) {
  9051. callback();
  9052. } else {
  9053. replenish();
  9054. }
  9055. }
  9056. });
  9057. }
  9058. })();
  9059. };
  9060. var retry = function (times, iterator, callback) {
  9061. var next = function (index) {
  9062. iterator(function (err, data) {
  9063. if (err && index < times) {
  9064. next(index + 1);
  9065. } else {
  9066. callback(err, data);
  9067. }
  9068. });
  9069. };
  9070. if (times < 1) {
  9071. callback();
  9072. } else {
  9073. next(1);
  9074. }
  9075. };
  9076. var async = {
  9077. eachLimit: eachLimit,
  9078. retry: retry
  9079. };
  9080. module.exports = async;
  9081. /***/ })
  9082. /******/ ]);
  9083. });