phar_object.c 166 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463
  1. /*
  2. +----------------------------------------------------------------------+
  3. | phar php single-file executable PHP extension |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 2005-2016 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt. |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Authors: Gregory Beaver <cellog@php.net> |
  16. | Marcus Boerger <helly@php.net> |
  17. +----------------------------------------------------------------------+
  18. */
  19. /* $Id$ */
  20. #include "phar_internal.h"
  21. #include "func_interceptors.h"
  22. static zend_class_entry *phar_ce_archive;
  23. static zend_class_entry *phar_ce_data;
  24. static zend_class_entry *phar_ce_PharException;
  25. #if HAVE_SPL
  26. static zend_class_entry *phar_ce_entry;
  27. #endif
  28. #if PHP_MAJOR_VERSION > 5 || ((PHP_MAJOR_VERSION == 5) && (PHP_MINOR_VERSION >= 3))
  29. # define PHAR_ARG_INFO
  30. #else
  31. # define PHAR_ARG_INFO static
  32. #endif
  33. static int phar_file_type(HashTable *mimes, char *file, char **mime_type TSRMLS_DC) /* {{{ */
  34. {
  35. char *ext;
  36. phar_mime_type *mime;
  37. ext = strrchr(file, '.');
  38. if (!ext) {
  39. *mime_type = "text/plain";
  40. /* no file extension = assume text/plain */
  41. return PHAR_MIME_OTHER;
  42. }
  43. ++ext;
  44. if (SUCCESS != zend_hash_find(mimes, ext, strlen(ext), (void **) &mime)) {
  45. *mime_type = "application/octet-stream";
  46. return PHAR_MIME_OTHER;
  47. }
  48. *mime_type = mime->mime;
  49. return mime->type;
  50. }
  51. /* }}} */
  52. static void phar_mung_server_vars(char *fname, char *entry, int entry_len, char *basename, int request_uri_len TSRMLS_DC) /* {{{ */
  53. {
  54. HashTable *_SERVER;
  55. zval **stuff;
  56. char *path_info;
  57. int basename_len = strlen(basename);
  58. int code;
  59. zval *temp;
  60. /* "tweak" $_SERVER variables requested in earlier call to Phar::mungServer() */
  61. if (!PG(http_globals)[TRACK_VARS_SERVER]) {
  62. return;
  63. }
  64. _SERVER = Z_ARRVAL_P(PG(http_globals)[TRACK_VARS_SERVER]);
  65. /* PATH_INFO and PATH_TRANSLATED should always be munged */
  66. if (SUCCESS == zend_hash_find(_SERVER, "PATH_INFO", sizeof("PATH_INFO"), (void **) &stuff)) {
  67. path_info = Z_STRVAL_PP(stuff);
  68. code = Z_STRLEN_PP(stuff);
  69. if (Z_STRLEN_PP(stuff) > entry_len && !memcmp(Z_STRVAL_PP(stuff), entry, entry_len)) {
  70. ZVAL_STRINGL(*stuff, Z_STRVAL_PP(stuff) + entry_len, request_uri_len, 1);
  71. MAKE_STD_ZVAL(temp);
  72. ZVAL_STRINGL(temp, path_info, code, 0);
  73. zend_hash_update(_SERVER, "PHAR_PATH_INFO", sizeof("PHAR_PATH_INFO"), &temp, sizeof(zval **), NULL);
  74. }
  75. }
  76. if (SUCCESS == zend_hash_find(_SERVER, "PATH_TRANSLATED", sizeof("PATH_TRANSLATED"), (void **) &stuff)) {
  77. path_info = Z_STRVAL_PP(stuff);
  78. code = Z_STRLEN_PP(stuff);
  79. Z_STRLEN_PP(stuff) = spprintf(&(Z_STRVAL_PP(stuff)), 4096, "phar://%s%s", fname, entry);
  80. MAKE_STD_ZVAL(temp);
  81. ZVAL_STRINGL(temp, path_info, code, 0);
  82. zend_hash_update(_SERVER, "PHAR_PATH_TRANSLATED", sizeof("PHAR_PATH_TRANSLATED"), (void *) &temp, sizeof(zval **), NULL);
  83. }
  84. if (!PHAR_GLOBALS->phar_SERVER_mung_list) {
  85. return;
  86. }
  87. if (PHAR_GLOBALS->phar_SERVER_mung_list & PHAR_MUNG_REQUEST_URI) {
  88. if (SUCCESS == zend_hash_find(_SERVER, "REQUEST_URI", sizeof("REQUEST_URI"), (void **) &stuff)) {
  89. path_info = Z_STRVAL_PP(stuff);
  90. code = Z_STRLEN_PP(stuff);
  91. if (Z_STRLEN_PP(stuff) > basename_len && !memcmp(Z_STRVAL_PP(stuff), basename, basename_len)) {
  92. ZVAL_STRINGL(*stuff, Z_STRVAL_PP(stuff) + basename_len, Z_STRLEN_PP(stuff) - basename_len, 1);
  93. MAKE_STD_ZVAL(temp);
  94. ZVAL_STRINGL(temp, path_info, code, 0);
  95. zend_hash_update(_SERVER, "PHAR_REQUEST_URI", sizeof("PHAR_REQUEST_URI"), (void *) &temp, sizeof(zval **), NULL);
  96. }
  97. }
  98. }
  99. if (PHAR_GLOBALS->phar_SERVER_mung_list & PHAR_MUNG_PHP_SELF) {
  100. if (SUCCESS == zend_hash_find(_SERVER, "PHP_SELF", sizeof("PHP_SELF"), (void **) &stuff)) {
  101. path_info = Z_STRVAL_PP(stuff);
  102. code = Z_STRLEN_PP(stuff);
  103. if (Z_STRLEN_PP(stuff) > basename_len && !memcmp(Z_STRVAL_PP(stuff), basename, basename_len)) {
  104. ZVAL_STRINGL(*stuff, Z_STRVAL_PP(stuff) + basename_len, Z_STRLEN_PP(stuff) - basename_len, 1);
  105. MAKE_STD_ZVAL(temp);
  106. ZVAL_STRINGL(temp, path_info, code, 0);
  107. zend_hash_update(_SERVER, "PHAR_PHP_SELF", sizeof("PHAR_PHP_SELF"), (void *) &temp, sizeof(zval **), NULL);
  108. }
  109. }
  110. }
  111. if (PHAR_GLOBALS->phar_SERVER_mung_list & PHAR_MUNG_SCRIPT_NAME) {
  112. if (SUCCESS == zend_hash_find(_SERVER, "SCRIPT_NAME", sizeof("SCRIPT_NAME"), (void **) &stuff)) {
  113. path_info = Z_STRVAL_PP(stuff);
  114. code = Z_STRLEN_PP(stuff);
  115. ZVAL_STRINGL(*stuff, entry, entry_len, 1);
  116. MAKE_STD_ZVAL(temp);
  117. ZVAL_STRINGL(temp, path_info, code, 0);
  118. zend_hash_update(_SERVER, "PHAR_SCRIPT_NAME", sizeof("PHAR_SCRIPT_NAME"), (void *) &temp, sizeof(zval **), NULL);
  119. }
  120. }
  121. if (PHAR_GLOBALS->phar_SERVER_mung_list & PHAR_MUNG_SCRIPT_FILENAME) {
  122. if (SUCCESS == zend_hash_find(_SERVER, "SCRIPT_FILENAME", sizeof("SCRIPT_FILENAME"), (void **) &stuff)) {
  123. path_info = Z_STRVAL_PP(stuff);
  124. code = Z_STRLEN_PP(stuff);
  125. Z_STRLEN_PP(stuff) = spprintf(&(Z_STRVAL_PP(stuff)), 4096, "phar://%s%s", fname, entry);
  126. MAKE_STD_ZVAL(temp);
  127. ZVAL_STRINGL(temp, path_info, code, 0);
  128. zend_hash_update(_SERVER, "PHAR_SCRIPT_FILENAME", sizeof("PHAR_SCRIPT_FILENAME"), (void *) &temp, sizeof(zval **), NULL);
  129. }
  130. }
  131. }
  132. /* }}} */
  133. static int phar_file_action(phar_archive_data *phar, phar_entry_info *info, char *mime_type, int code, char *entry, int entry_len, char *arch, char *basename, char *ru, int ru_len TSRMLS_DC) /* {{{ */
  134. {
  135. char *name = NULL, buf[8192];
  136. const char *cwd;
  137. zend_syntax_highlighter_ini syntax_highlighter_ini;
  138. sapi_header_line ctr = {0};
  139. size_t got;
  140. int dummy = 1, name_len;
  141. zend_file_handle file_handle;
  142. zend_op_array *new_op_array;
  143. zval *result = NULL;
  144. php_stream *fp;
  145. off_t position;
  146. switch (code) {
  147. case PHAR_MIME_PHPS:
  148. efree(basename);
  149. /* highlight source */
  150. if (entry[0] == '/') {
  151. name_len = spprintf(&name, 4096, "phar://%s%s", arch, entry);
  152. } else {
  153. name_len = spprintf(&name, 4096, "phar://%s/%s", arch, entry);
  154. }
  155. php_get_highlight_struct(&syntax_highlighter_ini);
  156. highlight_file(name, &syntax_highlighter_ini TSRMLS_CC);
  157. efree(name);
  158. #ifdef PHP_WIN32
  159. efree(arch);
  160. #endif
  161. zend_bailout();
  162. case PHAR_MIME_OTHER:
  163. /* send headers, output file contents */
  164. efree(basename);
  165. ctr.line_len = spprintf(&(ctr.line), 0, "Content-type: %s", mime_type);
  166. sapi_header_op(SAPI_HEADER_REPLACE, &ctr TSRMLS_CC);
  167. efree(ctr.line);
  168. ctr.line_len = spprintf(&(ctr.line), 0, "Content-length: %u", info->uncompressed_filesize);
  169. sapi_header_op(SAPI_HEADER_REPLACE, &ctr TSRMLS_CC);
  170. efree(ctr.line);
  171. if (FAILURE == sapi_send_headers(TSRMLS_C)) {
  172. zend_bailout();
  173. }
  174. /* prepare to output */
  175. fp = phar_get_efp(info, 1 TSRMLS_CC);
  176. if (!fp) {
  177. char *error;
  178. if (!phar_open_jit(phar, info, &error TSRMLS_CC)) {
  179. if (error) {
  180. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  181. efree(error);
  182. }
  183. return -1;
  184. }
  185. fp = phar_get_efp(info, 1 TSRMLS_CC);
  186. }
  187. position = 0;
  188. phar_seek_efp(info, 0, SEEK_SET, 0, 1 TSRMLS_CC);
  189. do {
  190. got = php_stream_read(fp, buf, MIN(8192, info->uncompressed_filesize - position));
  191. if (got > 0) {
  192. PHPWRITE(buf, got);
  193. position += got;
  194. if (position == (off_t) info->uncompressed_filesize) {
  195. break;
  196. }
  197. }
  198. } while (1);
  199. zend_bailout();
  200. case PHAR_MIME_PHP:
  201. if (basename) {
  202. phar_mung_server_vars(arch, entry, entry_len, basename, ru_len TSRMLS_CC);
  203. efree(basename);
  204. }
  205. if (entry[0] == '/') {
  206. name_len = spprintf(&name, 4096, "phar://%s%s", arch, entry);
  207. } else {
  208. name_len = spprintf(&name, 4096, "phar://%s/%s", arch, entry);
  209. }
  210. file_handle.type = ZEND_HANDLE_FILENAME;
  211. file_handle.handle.fd = 0;
  212. file_handle.filename = name;
  213. file_handle.opened_path = NULL;
  214. file_handle.free_filename = 0;
  215. PHAR_G(cwd) = NULL;
  216. PHAR_G(cwd_len) = 0;
  217. if (zend_hash_add(&EG(included_files), name, name_len+1, (void *)&dummy, sizeof(int), NULL) == SUCCESS) {
  218. if ((cwd = zend_memrchr(entry, '/', entry_len))) {
  219. PHAR_G(cwd_init) = 1;
  220. if (entry == cwd) {
  221. /* root directory */
  222. PHAR_G(cwd_len) = 0;
  223. PHAR_G(cwd) = NULL;
  224. } else if (entry[0] == '/') {
  225. PHAR_G(cwd_len) = cwd - (entry + 1);
  226. PHAR_G(cwd) = estrndup(entry + 1, PHAR_G(cwd_len));
  227. } else {
  228. PHAR_G(cwd_len) = cwd - entry;
  229. PHAR_G(cwd) = estrndup(entry, PHAR_G(cwd_len));
  230. }
  231. }
  232. new_op_array = zend_compile_file(&file_handle, ZEND_REQUIRE TSRMLS_CC);
  233. if (!new_op_array) {
  234. zend_hash_del(&EG(included_files), name, name_len+1);
  235. }
  236. zend_destroy_file_handle(&file_handle TSRMLS_CC);
  237. } else {
  238. efree(name);
  239. new_op_array = NULL;
  240. }
  241. #ifdef PHP_WIN32
  242. efree(arch);
  243. #endif
  244. if (new_op_array) {
  245. EG(return_value_ptr_ptr) = &result;
  246. EG(active_op_array) = new_op_array;
  247. zend_try {
  248. zend_execute(new_op_array TSRMLS_CC);
  249. if (PHAR_G(cwd)) {
  250. efree(PHAR_G(cwd));
  251. PHAR_G(cwd) = NULL;
  252. PHAR_G(cwd_len) = 0;
  253. }
  254. PHAR_G(cwd_init) = 0;
  255. efree(name);
  256. destroy_op_array(new_op_array TSRMLS_CC);
  257. efree(new_op_array);
  258. if (EG(return_value_ptr_ptr) && *EG(return_value_ptr_ptr)) {
  259. zval_ptr_dtor(EG(return_value_ptr_ptr));
  260. }
  261. } zend_catch {
  262. if (PHAR_G(cwd)) {
  263. efree(PHAR_G(cwd));
  264. PHAR_G(cwd) = NULL;
  265. PHAR_G(cwd_len) = 0;
  266. }
  267. PHAR_G(cwd_init) = 0;
  268. efree(name);
  269. } zend_end_try();
  270. zend_bailout();
  271. }
  272. return PHAR_MIME_PHP;
  273. }
  274. return -1;
  275. }
  276. /* }}} */
  277. static void phar_do_403(char *entry, int entry_len TSRMLS_DC) /* {{{ */
  278. {
  279. sapi_header_line ctr = {0};
  280. ctr.response_code = 403;
  281. ctr.line_len = sizeof("HTTP/1.0 403 Access Denied")-1;
  282. ctr.line = "HTTP/1.0 403 Access Denied";
  283. sapi_header_op(SAPI_HEADER_REPLACE, &ctr TSRMLS_CC);
  284. sapi_send_headers(TSRMLS_C);
  285. PHPWRITE("<html>\n <head>\n <title>Access Denied</title>\n </head>\n <body>\n <h1>403 - File ", sizeof("<html>\n <head>\n <title>Access Denied</title>\n </head>\n <body>\n <h1>403 - File ") - 1);
  286. PHPWRITE("Access Denied</h1>\n </body>\n</html>", sizeof("Access Denied</h1>\n </body>\n</html>") - 1);
  287. }
  288. /* }}} */
  289. static void phar_do_404(phar_archive_data *phar, char *fname, int fname_len, char *f404, int f404_len, char *entry, int entry_len TSRMLS_DC) /* {{{ */
  290. {
  291. sapi_header_line ctr = {0};
  292. phar_entry_info *info;
  293. if (phar && f404_len) {
  294. info = phar_get_entry_info(phar, f404, f404_len, NULL, 1 TSRMLS_CC);
  295. if (info) {
  296. phar_file_action(phar, info, "text/html", PHAR_MIME_PHP, f404, f404_len, fname, NULL, NULL, 0 TSRMLS_CC);
  297. return;
  298. }
  299. }
  300. ctr.response_code = 404;
  301. ctr.line_len = sizeof("HTTP/1.0 404 Not Found")-1;
  302. ctr.line = "HTTP/1.0 404 Not Found";
  303. sapi_header_op(SAPI_HEADER_REPLACE, &ctr TSRMLS_CC);
  304. sapi_send_headers(TSRMLS_C);
  305. PHPWRITE("<html>\n <head>\n <title>File Not Found</title>\n </head>\n <body>\n <h1>404 - File ", sizeof("<html>\n <head>\n <title>File Not Found</title>\n </head>\n <body>\n <h1>404 - File ") - 1);
  306. PHPWRITE("Not Found</h1>\n </body>\n</html>", sizeof("Not Found</h1>\n </body>\n</html>") - 1);
  307. }
  308. /* }}} */
  309. /* post-process REQUEST_URI and retrieve the actual request URI. This is for
  310. cases like http://localhost/blah.phar/path/to/file.php/extra/stuff
  311. which calls "blah.phar" file "path/to/file.php" with PATH_INFO "/extra/stuff" */
  312. static void phar_postprocess_ru_web(char *fname, int fname_len, char **entry, int *entry_len, char **ru, int *ru_len TSRMLS_DC) /* {{{ */
  313. {
  314. char *e = *entry + 1, *u = NULL, *u1 = NULL, *saveu = NULL;
  315. int e_len = *entry_len - 1, u_len = 0;
  316. phar_archive_data **pphar = NULL;
  317. /* we already know we can retrieve the phar if we reach here */
  318. zend_hash_find(&(PHAR_GLOBALS->phar_fname_map), fname, fname_len, (void **) &pphar);
  319. if (!pphar && PHAR_G(manifest_cached)) {
  320. zend_hash_find(&cached_phars, fname, fname_len, (void **) &pphar);
  321. }
  322. do {
  323. if (zend_hash_exists(&((*pphar)->manifest), e, e_len)) {
  324. if (u) {
  325. u[0] = '/';
  326. *ru = estrndup(u, u_len+1);
  327. ++u_len;
  328. u[0] = '\0';
  329. } else {
  330. *ru = NULL;
  331. }
  332. *ru_len = u_len;
  333. *entry_len = e_len + 1;
  334. return;
  335. }
  336. if (u) {
  337. u1 = strrchr(e, '/');
  338. u[0] = '/';
  339. saveu = u;
  340. e_len += u_len + 1;
  341. u = u1;
  342. if (!u) {
  343. return;
  344. }
  345. } else {
  346. u = strrchr(e, '/');
  347. if (!u) {
  348. if (saveu) {
  349. saveu[0] = '/';
  350. }
  351. return;
  352. }
  353. }
  354. u[0] = '\0';
  355. u_len = strlen(u + 1);
  356. e_len -= u_len + 1;
  357. if (e_len < 0) {
  358. if (saveu) {
  359. saveu[0] = '/';
  360. }
  361. return;
  362. }
  363. } while (1);
  364. }
  365. /* }}} */
  366. /* {{{ proto void Phar::running([bool retphar = true])
  367. * return the name of the currently running phar archive. If the optional parameter
  368. * is set to true, return the phar:// URL to the currently running phar
  369. */
  370. PHP_METHOD(Phar, running)
  371. {
  372. char *fname, *arch, *entry;
  373. int fname_len, arch_len, entry_len;
  374. zend_bool retphar = 1;
  375. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|b", &retphar) == FAILURE) {
  376. return;
  377. }
  378. fname = (char*)zend_get_executed_filename(TSRMLS_C);
  379. fname_len = strlen(fname);
  380. if (fname_len > 7 && !memcmp(fname, "phar://", 7) && SUCCESS == phar_split_fname(fname, fname_len, &arch, &arch_len, &entry, &entry_len, 2, 0 TSRMLS_CC)) {
  381. efree(entry);
  382. if (retphar) {
  383. RETVAL_STRINGL(fname, arch_len + 7, 1);
  384. efree(arch);
  385. return;
  386. } else {
  387. RETURN_STRINGL(arch, arch_len, 0);
  388. }
  389. }
  390. RETURN_STRINGL("", 0, 1);
  391. }
  392. /* }}} */
  393. /* {{{ proto void Phar::mount(string pharpath, string externalfile)
  394. * mount an external file or path to a location within the phar. This maps
  395. * an external file or directory to a location within the phar archive, allowing
  396. * reference to an external location as if it were within the phar archive. This
  397. * is useful for writable temp files like databases
  398. */
  399. PHP_METHOD(Phar, mount)
  400. {
  401. char *fname, *arch = NULL, *entry = NULL, *path, *actual;
  402. int fname_len, arch_len, entry_len, path_len, actual_len;
  403. phar_archive_data **pphar;
  404. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "pp", &path, &path_len, &actual, &actual_len) == FAILURE) {
  405. return;
  406. }
  407. fname = (char*)zend_get_executed_filename(TSRMLS_C);
  408. fname_len = strlen(fname);
  409. #ifdef PHP_WIN32
  410. phar_unixify_path_separators(fname, fname_len);
  411. #endif
  412. if (fname_len > 7 && !memcmp(fname, "phar://", 7) && SUCCESS == phar_split_fname(fname, fname_len, &arch, &arch_len, &entry, &entry_len, 2, 0 TSRMLS_CC)) {
  413. efree(entry);
  414. entry = NULL;
  415. if (path_len > 7 && !memcmp(path, "phar://", 7)) {
  416. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "Can only mount internal paths within a phar archive, use a relative path instead of \"%s\"", path);
  417. efree(arch);
  418. return;
  419. }
  420. carry_on2:
  421. if (SUCCESS != zend_hash_find(&(PHAR_GLOBALS->phar_fname_map), arch, arch_len, (void **)&pphar)) {
  422. if (PHAR_G(manifest_cached) && SUCCESS == zend_hash_find(&cached_phars, arch, arch_len, (void **)&pphar)) {
  423. if (SUCCESS == phar_copy_on_write(pphar TSRMLS_CC)) {
  424. goto carry_on;
  425. }
  426. }
  427. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s is not a phar archive, cannot mount", arch);
  428. if (arch) {
  429. efree(arch);
  430. }
  431. return;
  432. }
  433. carry_on:
  434. if (SUCCESS != phar_mount_entry(*pphar, actual, actual_len, path, path_len TSRMLS_CC)) {
  435. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "Mounting of %s to %s within phar %s failed", path, actual, arch);
  436. if (path && path == entry) {
  437. efree(entry);
  438. }
  439. if (arch) {
  440. efree(arch);
  441. }
  442. return;
  443. }
  444. if (entry && path && path == entry) {
  445. efree(entry);
  446. }
  447. if (arch) {
  448. efree(arch);
  449. }
  450. return;
  451. } else if (PHAR_GLOBALS->phar_fname_map.arBuckets && SUCCESS == zend_hash_find(&(PHAR_GLOBALS->phar_fname_map), fname, fname_len, (void **)&pphar)) {
  452. goto carry_on;
  453. } else if (PHAR_G(manifest_cached) && SUCCESS == zend_hash_find(&cached_phars, fname, fname_len, (void **)&pphar)) {
  454. if (SUCCESS == phar_copy_on_write(pphar TSRMLS_CC)) {
  455. goto carry_on;
  456. }
  457. goto carry_on;
  458. } else if (SUCCESS == phar_split_fname(path, path_len, &arch, &arch_len, &entry, &entry_len, 2, 0 TSRMLS_CC)) {
  459. path = entry;
  460. path_len = entry_len;
  461. goto carry_on2;
  462. }
  463. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "Mounting of %s to %s failed", path, actual);
  464. }
  465. /* }}} */
  466. /* {{{ proto void Phar::webPhar([string alias, [string index, [string f404, [array mimetypes, [callback rewrites]]]]])
  467. * mapPhar for web-based phars. Reads the currently executed file (a phar)
  468. * and registers its manifest. When executed in the CLI or CGI command-line sapi,
  469. * this works exactly like mapPhar(). When executed by a web-based sapi, this
  470. * reads $_SERVER['REQUEST_URI'] (the actual original value) and parses out the
  471. * intended internal file.
  472. */
  473. PHP_METHOD(Phar, webPhar)
  474. {
  475. zval *mimeoverride = NULL, *rewrite = NULL;
  476. char *alias = NULL, *error, *index_php = NULL, *f404 = NULL, *ru = NULL;
  477. int alias_len = 0, ret, f404_len = 0, free_pathinfo = 0, ru_len = 0;
  478. char *fname, *path_info, *mime_type = NULL, *entry, *pt;
  479. const char *basename;
  480. int fname_len, entry_len, code, index_php_len = 0, not_cgi;
  481. phar_archive_data *phar = NULL;
  482. phar_entry_info *info = NULL;
  483. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|s!s!saz", &alias, &alias_len, &index_php, &index_php_len, &f404, &f404_len, &mimeoverride, &rewrite) == FAILURE) {
  484. return;
  485. }
  486. phar_request_initialize(TSRMLS_C);
  487. fname = (char*)zend_get_executed_filename(TSRMLS_C);
  488. fname_len = strlen(fname);
  489. if (phar_open_executed_filename(alias, alias_len, &error TSRMLS_CC) != SUCCESS) {
  490. if (error) {
  491. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  492. efree(error);
  493. }
  494. return;
  495. }
  496. /* retrieve requested file within phar */
  497. if (!(SG(request_info).request_method && SG(request_info).request_uri && (!strcmp(SG(request_info).request_method, "GET") || !strcmp(SG(request_info).request_method, "POST")))) {
  498. return;
  499. }
  500. #ifdef PHP_WIN32
  501. fname = estrndup(fname, fname_len);
  502. phar_unixify_path_separators(fname, fname_len);
  503. #endif
  504. basename = zend_memrchr(fname, '/', fname_len);
  505. if (!basename) {
  506. basename = fname;
  507. } else {
  508. ++basename;
  509. }
  510. if ((strlen(sapi_module.name) == sizeof("cgi-fcgi")-1 && !strncmp(sapi_module.name, "cgi-fcgi", sizeof("cgi-fcgi")-1))
  511. || (strlen(sapi_module.name) == sizeof("fpm-fcgi")-1 && !strncmp(sapi_module.name, "fpm-fcgi", sizeof("fpm-fcgi")-1))
  512. || (strlen(sapi_module.name) == sizeof("cgi")-1 && !strncmp(sapi_module.name, "cgi", sizeof("cgi")-1))) {
  513. if (PG(http_globals)[TRACK_VARS_SERVER]) {
  514. HashTable *_server = Z_ARRVAL_P(PG(http_globals)[TRACK_VARS_SERVER]);
  515. zval **z_script_name, **z_path_info;
  516. if (SUCCESS != zend_hash_find(_server, "SCRIPT_NAME", sizeof("SCRIPT_NAME"), (void**)&z_script_name) ||
  517. IS_STRING != Z_TYPE_PP(z_script_name) ||
  518. !strstr(Z_STRVAL_PP(z_script_name), basename)) {
  519. return;
  520. }
  521. if (SUCCESS == zend_hash_find(_server, "PATH_INFO", sizeof("PATH_INFO"), (void**)&z_path_info) &&
  522. IS_STRING == Z_TYPE_PP(z_path_info)) {
  523. entry_len = Z_STRLEN_PP(z_path_info);
  524. entry = estrndup(Z_STRVAL_PP(z_path_info), entry_len);
  525. path_info = emalloc(Z_STRLEN_PP(z_script_name) + entry_len + 1);
  526. memcpy(path_info, Z_STRVAL_PP(z_script_name), Z_STRLEN_PP(z_script_name));
  527. memcpy(path_info + Z_STRLEN_PP(z_script_name), entry, entry_len + 1);
  528. free_pathinfo = 1;
  529. } else {
  530. entry_len = 0;
  531. entry = estrndup("", 0);
  532. path_info = Z_STRVAL_PP(z_script_name);
  533. }
  534. pt = estrndup(Z_STRVAL_PP(z_script_name), Z_STRLEN_PP(z_script_name));
  535. } else {
  536. char *testit;
  537. testit = sapi_getenv("SCRIPT_NAME", sizeof("SCRIPT_NAME")-1 TSRMLS_CC);
  538. if (!(pt = strstr(testit, basename))) {
  539. efree(testit);
  540. return;
  541. }
  542. path_info = sapi_getenv("PATH_INFO", sizeof("PATH_INFO")-1 TSRMLS_CC);
  543. if (path_info) {
  544. entry = path_info;
  545. entry_len = strlen(entry);
  546. spprintf(&path_info, 0, "%s%s", testit, path_info);
  547. free_pathinfo = 1;
  548. } else {
  549. path_info = testit;
  550. free_pathinfo = 1;
  551. entry = estrndup("", 0);
  552. entry_len = 0;
  553. }
  554. pt = estrndup(testit, (pt - testit) + (fname_len - (basename - fname)));
  555. }
  556. not_cgi = 0;
  557. } else {
  558. path_info = SG(request_info).request_uri;
  559. if (!(pt = strstr(path_info, basename))) {
  560. /* this can happen with rewrite rules - and we have no idea what to do then, so return */
  561. return;
  562. }
  563. entry_len = strlen(path_info);
  564. entry_len -= (pt - path_info) + (fname_len - (basename - fname));
  565. entry = estrndup(pt + (fname_len - (basename - fname)), entry_len);
  566. pt = estrndup(path_info, (pt - path_info) + (fname_len - (basename - fname)));
  567. not_cgi = 1;
  568. }
  569. if (rewrite) {
  570. zend_fcall_info fci;
  571. zend_fcall_info_cache fcc;
  572. zval *params, *retval_ptr, **zp[1];
  573. MAKE_STD_ZVAL(params);
  574. ZVAL_STRINGL(params, entry, entry_len, 1);
  575. zp[0] = &params;
  576. if (FAILURE == zend_fcall_info_init(rewrite, 0, &fci, &fcc, NULL, NULL TSRMLS_CC)) {
  577. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar error: invalid rewrite callback");
  578. if (free_pathinfo) {
  579. efree(path_info);
  580. }
  581. return;
  582. }
  583. fci.param_count = 1;
  584. fci.params = zp;
  585. Z_ADDREF_P(params);
  586. fci.retval_ptr_ptr = &retval_ptr;
  587. if (FAILURE == zend_call_function(&fci, &fcc TSRMLS_CC)) {
  588. if (!EG(exception)) {
  589. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar error: failed to call rewrite callback");
  590. }
  591. if (free_pathinfo) {
  592. efree(path_info);
  593. }
  594. return;
  595. }
  596. if (!fci.retval_ptr_ptr || !retval_ptr) {
  597. if (free_pathinfo) {
  598. efree(path_info);
  599. }
  600. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar error: rewrite callback must return a string or false");
  601. return;
  602. }
  603. switch (Z_TYPE_P(retval_ptr)) {
  604. case IS_STRING:
  605. efree(entry);
  606. if (fci.retval_ptr_ptr != &retval_ptr) {
  607. entry = estrndup(Z_STRVAL_PP(fci.retval_ptr_ptr), Z_STRLEN_PP(fci.retval_ptr_ptr));
  608. entry_len = Z_STRLEN_PP(fci.retval_ptr_ptr);
  609. } else {
  610. entry = Z_STRVAL_P(retval_ptr);
  611. entry_len = Z_STRLEN_P(retval_ptr);
  612. }
  613. break;
  614. case IS_BOOL:
  615. phar_do_403(entry, entry_len TSRMLS_CC);
  616. if (free_pathinfo) {
  617. efree(path_info);
  618. }
  619. zend_bailout();
  620. return;
  621. default:
  622. efree(retval_ptr);
  623. if (free_pathinfo) {
  624. efree(path_info);
  625. }
  626. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar error: rewrite callback must return a string or false");
  627. return;
  628. }
  629. }
  630. if (entry_len) {
  631. phar_postprocess_ru_web(fname, fname_len, &entry, &entry_len, &ru, &ru_len TSRMLS_CC);
  632. }
  633. if (!entry_len || (entry_len == 1 && entry[0] == '/')) {
  634. efree(entry);
  635. /* direct request */
  636. if (index_php_len) {
  637. entry = index_php;
  638. entry_len = index_php_len;
  639. if (entry[0] != '/') {
  640. spprintf(&entry, 0, "/%s", index_php);
  641. ++entry_len;
  642. }
  643. } else {
  644. /* assume "index.php" is starting point */
  645. entry = estrndup("/index.php", sizeof("/index.php"));
  646. entry_len = sizeof("/index.php")-1;
  647. }
  648. if (FAILURE == phar_get_archive(&phar, fname, fname_len, NULL, 0, NULL TSRMLS_CC) ||
  649. (info = phar_get_entry_info(phar, entry, entry_len, NULL, 0 TSRMLS_CC)) == NULL) {
  650. phar_do_404(phar, fname, fname_len, f404, f404_len, entry, entry_len TSRMLS_CC);
  651. if (free_pathinfo) {
  652. efree(path_info);
  653. }
  654. zend_bailout();
  655. } else {
  656. char *tmp = NULL, sa = '\0';
  657. sapi_header_line ctr = {0};
  658. ctr.response_code = 301;
  659. ctr.line_len = sizeof("HTTP/1.1 301 Moved Permanently")-1;
  660. ctr.line = "HTTP/1.1 301 Moved Permanently";
  661. sapi_header_op(SAPI_HEADER_REPLACE, &ctr TSRMLS_CC);
  662. if (not_cgi) {
  663. tmp = strstr(path_info, basename) + fname_len;
  664. sa = *tmp;
  665. *tmp = '\0';
  666. }
  667. ctr.response_code = 0;
  668. if (path_info[strlen(path_info)-1] == '/') {
  669. ctr.line_len = spprintf(&(ctr.line), 4096, "Location: %s%s", path_info, entry + 1);
  670. } else {
  671. ctr.line_len = spprintf(&(ctr.line), 4096, "Location: %s%s", path_info, entry);
  672. }
  673. if (not_cgi) {
  674. *tmp = sa;
  675. }
  676. if (free_pathinfo) {
  677. efree(path_info);
  678. }
  679. sapi_header_op(SAPI_HEADER_REPLACE, &ctr TSRMLS_CC);
  680. sapi_send_headers(TSRMLS_C);
  681. efree(ctr.line);
  682. zend_bailout();
  683. }
  684. }
  685. if (FAILURE == phar_get_archive(&phar, fname, fname_len, NULL, 0, NULL TSRMLS_CC) ||
  686. (info = phar_get_entry_info(phar, entry, entry_len, NULL, 0 TSRMLS_CC)) == NULL) {
  687. phar_do_404(phar, fname, fname_len, f404, f404_len, entry, entry_len TSRMLS_CC);
  688. #ifdef PHP_WIN32
  689. efree(fname);
  690. #endif
  691. zend_bailout();
  692. }
  693. if (mimeoverride && zend_hash_num_elements(Z_ARRVAL_P(mimeoverride))) {
  694. const char *ext = zend_memrchr(entry, '.', entry_len);
  695. zval **val;
  696. if (ext) {
  697. ++ext;
  698. if (SUCCESS == zend_hash_find(Z_ARRVAL_P(mimeoverride), ext, strlen(ext)+1, (void **) &val)) {
  699. switch (Z_TYPE_PP(val)) {
  700. case IS_LONG:
  701. if (Z_LVAL_PP(val) == PHAR_MIME_PHP || Z_LVAL_PP(val) == PHAR_MIME_PHPS) {
  702. mime_type = "";
  703. code = Z_LVAL_PP(val);
  704. } else {
  705. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "Unknown mime type specifier used, only Phar::PHP, Phar::PHPS and a mime type string are allowed");
  706. #ifdef PHP_WIN32
  707. efree(fname);
  708. #endif
  709. RETURN_FALSE;
  710. }
  711. break;
  712. case IS_STRING:
  713. mime_type = Z_STRVAL_PP(val);
  714. code = PHAR_MIME_OTHER;
  715. break;
  716. default:
  717. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "Unknown mime type specifier used (not a string or int), only Phar::PHP, Phar::PHPS and a mime type string are allowed");
  718. #ifdef PHP_WIN32
  719. efree(fname);
  720. #endif
  721. RETURN_FALSE;
  722. }
  723. }
  724. }
  725. }
  726. if (!mime_type) {
  727. code = phar_file_type(&PHAR_G(mime_types), entry, &mime_type TSRMLS_CC);
  728. }
  729. ret = phar_file_action(phar, info, mime_type, code, entry, entry_len, fname, pt, ru, ru_len TSRMLS_CC);
  730. }
  731. /* }}} */
  732. /* {{{ proto void Phar::mungServer(array munglist)
  733. * Defines a list of up to 4 $_SERVER variables that should be modified for execution
  734. * to mask the presence of the phar archive. This should be used in conjunction with
  735. * Phar::webPhar(), and has no effect otherwise
  736. * SCRIPT_NAME, PHP_SELF, REQUEST_URI and SCRIPT_FILENAME
  737. */
  738. PHP_METHOD(Phar, mungServer)
  739. {
  740. zval *mungvalues;
  741. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "a", &mungvalues) == FAILURE) {
  742. return;
  743. }
  744. if (!zend_hash_num_elements(Z_ARRVAL_P(mungvalues))) {
  745. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "No values passed to Phar::mungServer(), expecting an array of any of these strings: PHP_SELF, REQUEST_URI, SCRIPT_FILENAME, SCRIPT_NAME");
  746. return;
  747. }
  748. if (zend_hash_num_elements(Z_ARRVAL_P(mungvalues)) > 4) {
  749. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "Too many values passed to Phar::mungServer(), expecting an array of any of these strings: PHP_SELF, REQUEST_URI, SCRIPT_FILENAME, SCRIPT_NAME");
  750. return;
  751. }
  752. phar_request_initialize(TSRMLS_C);
  753. for (zend_hash_internal_pointer_reset(Z_ARRVAL_P(mungvalues)); SUCCESS == zend_hash_has_more_elements(Z_ARRVAL_P(mungvalues)); zend_hash_move_forward(Z_ARRVAL_P(mungvalues))) {
  754. zval **data = NULL;
  755. if (SUCCESS != zend_hash_get_current_data(Z_ARRVAL_P(mungvalues), (void **) &data)) {
  756. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "unable to retrieve array value in Phar::mungServer()");
  757. return;
  758. }
  759. if (Z_TYPE_PP(data) != IS_STRING) {
  760. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "Non-string value passed to Phar::mungServer(), expecting an array of any of these strings: PHP_SELF, REQUEST_URI, SCRIPT_FILENAME, SCRIPT_NAME");
  761. return;
  762. }
  763. if (Z_STRLEN_PP(data) == sizeof("PHP_SELF")-1 && !strncmp(Z_STRVAL_PP(data), "PHP_SELF", sizeof("PHP_SELF")-1)) {
  764. PHAR_GLOBALS->phar_SERVER_mung_list |= PHAR_MUNG_PHP_SELF;
  765. }
  766. if (Z_STRLEN_PP(data) == sizeof("REQUEST_URI")-1) {
  767. if (!strncmp(Z_STRVAL_PP(data), "REQUEST_URI", sizeof("REQUEST_URI")-1)) {
  768. PHAR_GLOBALS->phar_SERVER_mung_list |= PHAR_MUNG_REQUEST_URI;
  769. }
  770. if (!strncmp(Z_STRVAL_PP(data), "SCRIPT_NAME", sizeof("SCRIPT_NAME")-1)) {
  771. PHAR_GLOBALS->phar_SERVER_mung_list |= PHAR_MUNG_SCRIPT_NAME;
  772. }
  773. }
  774. if (Z_STRLEN_PP(data) == sizeof("SCRIPT_FILENAME")-1 && !strncmp(Z_STRVAL_PP(data), "SCRIPT_FILENAME", sizeof("SCRIPT_FILENAME")-1)) {
  775. PHAR_GLOBALS->phar_SERVER_mung_list |= PHAR_MUNG_SCRIPT_FILENAME;
  776. }
  777. }
  778. }
  779. /* }}} */
  780. /* {{{ proto void Phar::interceptFileFuncs()
  781. * instructs phar to intercept fopen, file_get_contents, opendir, and all of the stat-related functions
  782. * and return stat on files within the phar for relative paths
  783. *
  784. * Once called, this cannot be reversed, and continue until the end of the request.
  785. *
  786. * This allows legacy scripts to be pharred unmodified
  787. */
  788. PHP_METHOD(Phar, interceptFileFuncs)
  789. {
  790. if (zend_parse_parameters_none() == FAILURE) {
  791. return;
  792. }
  793. phar_intercept_functions(TSRMLS_C);
  794. }
  795. /* }}} */
  796. /* {{{ proto array Phar::createDefaultStub([string indexfile[, string webindexfile]])
  797. * Return a stub that can be used to run a phar-based archive without the phar extension
  798. * indexfile is the CLI startup filename, which defaults to "index.php", webindexfile
  799. * is the web startup filename, and also defaults to "index.php"
  800. */
  801. PHP_METHOD(Phar, createDefaultStub)
  802. {
  803. char *index = NULL, *webindex = NULL, *stub, *error;
  804. int index_len = 0, webindex_len = 0;
  805. size_t stub_len;
  806. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|pp", &index, &index_len, &webindex, &webindex_len) == FAILURE) {
  807. return;
  808. }
  809. stub = phar_create_default_stub(index, webindex, &stub_len, &error TSRMLS_CC);
  810. if (error) {
  811. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  812. efree(error);
  813. return;
  814. }
  815. RETURN_STRINGL(stub, stub_len, 0);
  816. }
  817. /* }}} */
  818. /* {{{ proto mixed Phar::mapPhar([string alias, [int dataoffset]])
  819. * Reads the currently executed file (a phar) and registers its manifest */
  820. PHP_METHOD(Phar, mapPhar)
  821. {
  822. char *alias = NULL, *error;
  823. int alias_len = 0;
  824. long dataoffset = 0;
  825. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|s!l", &alias, &alias_len, &dataoffset) == FAILURE) {
  826. return;
  827. }
  828. phar_request_initialize(TSRMLS_C);
  829. RETVAL_BOOL(phar_open_executed_filename(alias, alias_len, &error TSRMLS_CC) == SUCCESS);
  830. if (error) {
  831. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  832. efree(error);
  833. }
  834. } /* }}} */
  835. /* {{{ proto mixed Phar::loadPhar(string filename [, string alias])
  836. * Loads any phar archive with an alias */
  837. PHP_METHOD(Phar, loadPhar)
  838. {
  839. char *fname, *alias = NULL, *error;
  840. int fname_len, alias_len = 0;
  841. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p|s!", &fname, &fname_len, &alias, &alias_len) == FAILURE) {
  842. return;
  843. }
  844. phar_request_initialize(TSRMLS_C);
  845. RETVAL_BOOL(phar_open_from_filename(fname, fname_len, alias, alias_len, REPORT_ERRORS, NULL, &error TSRMLS_CC) == SUCCESS);
  846. if (error) {
  847. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  848. efree(error);
  849. }
  850. } /* }}} */
  851. /* {{{ proto string Phar::apiVersion()
  852. * Returns the api version */
  853. PHP_METHOD(Phar, apiVersion)
  854. {
  855. if (zend_parse_parameters_none() == FAILURE) {
  856. return;
  857. }
  858. RETURN_STRINGL(PHP_PHAR_API_VERSION, sizeof(PHP_PHAR_API_VERSION)-1, 1);
  859. }
  860. /* }}}*/
  861. /* {{{ proto bool Phar::canCompress([int method])
  862. * Returns whether phar extension supports compression using zlib/bzip2 */
  863. PHP_METHOD(Phar, canCompress)
  864. {
  865. long method = 0;
  866. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|l", &method) == FAILURE) {
  867. return;
  868. }
  869. phar_request_initialize(TSRMLS_C);
  870. switch (method) {
  871. case PHAR_ENT_COMPRESSED_GZ:
  872. if (PHAR_G(has_zlib)) {
  873. RETURN_TRUE;
  874. } else {
  875. RETURN_FALSE;
  876. }
  877. case PHAR_ENT_COMPRESSED_BZ2:
  878. if (PHAR_G(has_bz2)) {
  879. RETURN_TRUE;
  880. } else {
  881. RETURN_FALSE;
  882. }
  883. default:
  884. if (PHAR_G(has_zlib) || PHAR_G(has_bz2)) {
  885. RETURN_TRUE;
  886. } else {
  887. RETURN_FALSE;
  888. }
  889. }
  890. }
  891. /* }}} */
  892. /* {{{ proto bool Phar::canWrite()
  893. * Returns whether phar extension supports writing and creating phars */
  894. PHP_METHOD(Phar, canWrite)
  895. {
  896. if (zend_parse_parameters_none() == FAILURE) {
  897. return;
  898. }
  899. RETURN_BOOL(!PHAR_G(readonly));
  900. }
  901. /* }}} */
  902. /* {{{ proto bool Phar::isValidPharFilename(string filename[, bool executable = true])
  903. * Returns whether the given filename is a valid phar filename */
  904. PHP_METHOD(Phar, isValidPharFilename)
  905. {
  906. char *fname;
  907. const char *ext_str;
  908. int fname_len, ext_len, is_executable;
  909. zend_bool executable = 1;
  910. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p|b", &fname, &fname_len, &executable) == FAILURE) {
  911. return;
  912. }
  913. is_executable = executable;
  914. RETVAL_BOOL(phar_detect_phar_fname_ext(fname, fname_len, &ext_str, &ext_len, is_executable, 2, 1 TSRMLS_CC) == SUCCESS);
  915. }
  916. /* }}} */
  917. #if HAVE_SPL
  918. /**
  919. * from spl_directory
  920. */
  921. static void phar_spl_foreign_dtor(spl_filesystem_object *object TSRMLS_DC) /* {{{ */
  922. {
  923. phar_archive_data *phar = (phar_archive_data *) object->oth;
  924. if (!phar->is_persistent) {
  925. phar_archive_delref(phar TSRMLS_CC);
  926. }
  927. object->oth = NULL;
  928. }
  929. /* }}} */
  930. /**
  931. * from spl_directory
  932. */
  933. static void phar_spl_foreign_clone(spl_filesystem_object *src, spl_filesystem_object *dst TSRMLS_DC) /* {{{ */
  934. {
  935. phar_archive_data *phar_data = (phar_archive_data *) dst->oth;
  936. if (!phar_data->is_persistent) {
  937. ++(phar_data->refcount);
  938. }
  939. }
  940. /* }}} */
  941. static spl_other_handler phar_spl_foreign_handler = {
  942. phar_spl_foreign_dtor,
  943. phar_spl_foreign_clone
  944. };
  945. #endif /* HAVE_SPL */
  946. /* {{{ proto void Phar::__construct(string fname [, int flags [, string alias]])
  947. * Construct a Phar archive object
  948. *
  949. * proto void PharData::__construct(string fname [[, int flags [, string alias]], int file format = Phar::TAR])
  950. * Construct a PharData archive object
  951. *
  952. * This function is used as the constructor for both the Phar and PharData
  953. * classes, hence the two prototypes above.
  954. */
  955. PHP_METHOD(Phar, __construct)
  956. {
  957. #if !HAVE_SPL
  958. zend_throw_exception_ex(zend_exception_get_default(TSRMLS_C), 0 TSRMLS_CC, "Cannot instantiate Phar object without SPL extension");
  959. #else
  960. char *fname, *alias = NULL, *error, *arch = NULL, *entry = NULL, *save_fname;
  961. int fname_len, alias_len = 0, arch_len, entry_len, is_data;
  962. long flags = SPL_FILE_DIR_SKIPDOTS|SPL_FILE_DIR_UNIXPATHS;
  963. long format = 0;
  964. phar_archive_object *phar_obj;
  965. phar_archive_data *phar_data;
  966. zval *zobj = getThis(), arg1, arg2;
  967. phar_obj = (phar_archive_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
  968. is_data = instanceof_function(Z_OBJCE_P(zobj), phar_ce_data TSRMLS_CC);
  969. if (is_data) {
  970. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p|ls!l", &fname, &fname_len, &flags, &alias, &alias_len, &format) == FAILURE) {
  971. return;
  972. }
  973. } else {
  974. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p|ls!", &fname, &fname_len, &flags, &alias, &alias_len) == FAILURE) {
  975. return;
  976. }
  977. }
  978. if (phar_obj->arc.archive) {
  979. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Cannot call constructor twice");
  980. return;
  981. }
  982. save_fname = fname;
  983. if (SUCCESS == phar_split_fname(fname, fname_len, &arch, &arch_len, &entry, &entry_len, !is_data, 2 TSRMLS_CC)) {
  984. /* use arch (the basename for the archive) for fname instead of fname */
  985. /* this allows support for RecursiveDirectoryIterator of subdirectories */
  986. #ifdef PHP_WIN32
  987. phar_unixify_path_separators(arch, arch_len);
  988. #endif
  989. fname = arch;
  990. fname_len = arch_len;
  991. #ifdef PHP_WIN32
  992. } else {
  993. arch = estrndup(fname, fname_len);
  994. arch_len = fname_len;
  995. fname = arch;
  996. phar_unixify_path_separators(arch, arch_len);
  997. #endif
  998. }
  999. if (phar_open_or_create_filename(fname, fname_len, alias, alias_len, is_data, REPORT_ERRORS, &phar_data, &error TSRMLS_CC) == FAILURE) {
  1000. if (fname == arch && fname != save_fname) {
  1001. efree(arch);
  1002. fname = save_fname;
  1003. }
  1004. if (entry) {
  1005. efree(entry);
  1006. }
  1007. if (error) {
  1008. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  1009. "%s", error);
  1010. efree(error);
  1011. } else {
  1012. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  1013. "Phar creation or opening failed");
  1014. }
  1015. return;
  1016. }
  1017. if (is_data && phar_data->is_tar && phar_data->is_brandnew && format == PHAR_FORMAT_ZIP) {
  1018. phar_data->is_zip = 1;
  1019. phar_data->is_tar = 0;
  1020. }
  1021. if (fname == arch) {
  1022. efree(arch);
  1023. fname = save_fname;
  1024. }
  1025. if ((is_data && !phar_data->is_data) || (!is_data && phar_data->is_data)) {
  1026. if (is_data) {
  1027. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  1028. "PharData class can only be used for non-executable tar and zip archives");
  1029. } else {
  1030. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  1031. "Phar class can only be used for executable tar and zip archives");
  1032. }
  1033. efree(entry);
  1034. return;
  1035. }
  1036. is_data = phar_data->is_data;
  1037. if (!phar_data->is_persistent) {
  1038. ++(phar_data->refcount);
  1039. }
  1040. phar_obj->arc.archive = phar_data;
  1041. phar_obj->spl.oth_handler = &phar_spl_foreign_handler;
  1042. if (entry) {
  1043. fname_len = spprintf(&fname, 0, "phar://%s%s", phar_data->fname, entry);
  1044. efree(entry);
  1045. } else {
  1046. fname_len = spprintf(&fname, 0, "phar://%s", phar_data->fname);
  1047. }
  1048. INIT_PZVAL(&arg1);
  1049. ZVAL_STRINGL(&arg1, fname, fname_len, 0);
  1050. INIT_PZVAL(&arg2);
  1051. ZVAL_LONG(&arg2, flags);
  1052. zend_call_method_with_2_params(&zobj, Z_OBJCE_P(zobj),
  1053. &spl_ce_RecursiveDirectoryIterator->constructor, "__construct", NULL, &arg1, &arg2);
  1054. if (!phar_data->is_persistent) {
  1055. phar_obj->arc.archive->is_data = is_data;
  1056. } else if (!EG(exception)) {
  1057. /* register this guy so we can modify if necessary */
  1058. zend_hash_add(&PHAR_GLOBALS->phar_persist_map, (const char *) phar_obj->arc.archive, sizeof(phar_obj->arc.archive), (void *) &phar_obj, sizeof(phar_archive_object **), NULL);
  1059. }
  1060. phar_obj->spl.info_class = phar_ce_entry;
  1061. efree(fname);
  1062. #endif /* HAVE_SPL */
  1063. }
  1064. /* }}} */
  1065. /* {{{ proto array Phar::getSupportedSignatures()
  1066. * Return array of supported signature types
  1067. */
  1068. PHP_METHOD(Phar, getSupportedSignatures)
  1069. {
  1070. if (zend_parse_parameters_none() == FAILURE) {
  1071. return;
  1072. }
  1073. array_init(return_value);
  1074. add_next_index_stringl(return_value, "MD5", 3, 1);
  1075. add_next_index_stringl(return_value, "SHA-1", 5, 1);
  1076. #ifdef PHAR_HASH_OK
  1077. add_next_index_stringl(return_value, "SHA-256", 7, 1);
  1078. add_next_index_stringl(return_value, "SHA-512", 7, 1);
  1079. #endif
  1080. #if PHAR_HAVE_OPENSSL
  1081. add_next_index_stringl(return_value, "OpenSSL", 7, 1);
  1082. #else
  1083. if (zend_hash_exists(&module_registry, "openssl", sizeof("openssl"))) {
  1084. add_next_index_stringl(return_value, "OpenSSL", 7, 1);
  1085. }
  1086. #endif
  1087. }
  1088. /* }}} */
  1089. /* {{{ proto array Phar::getSupportedCompression()
  1090. * Return array of supported comparession algorithms
  1091. */
  1092. PHP_METHOD(Phar, getSupportedCompression)
  1093. {
  1094. if (zend_parse_parameters_none() == FAILURE) {
  1095. return;
  1096. }
  1097. array_init(return_value);
  1098. phar_request_initialize(TSRMLS_C);
  1099. if (PHAR_G(has_zlib)) {
  1100. add_next_index_stringl(return_value, "GZ", 2, 1);
  1101. }
  1102. if (PHAR_G(has_bz2)) {
  1103. add_next_index_stringl(return_value, "BZIP2", 5, 1);
  1104. }
  1105. }
  1106. /* }}} */
  1107. /* {{{ proto array Phar::unlinkArchive(string archive)
  1108. * Completely remove a phar archive from memory and disk
  1109. */
  1110. PHP_METHOD(Phar, unlinkArchive)
  1111. {
  1112. char *fname, *error, *zname, *arch, *entry;
  1113. int fname_len, zname_len, arch_len, entry_len;
  1114. phar_archive_data *phar;
  1115. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p", &fname, &fname_len) == FAILURE) {
  1116. RETURN_FALSE;
  1117. }
  1118. if (!fname_len) {
  1119. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "Unknown phar archive \"\"");
  1120. return;
  1121. }
  1122. if (FAILURE == phar_open_from_filename(fname, fname_len, NULL, 0, REPORT_ERRORS, &phar, &error TSRMLS_CC)) {
  1123. if (error) {
  1124. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "Unknown phar archive \"%s\": %s", fname, error);
  1125. efree(error);
  1126. } else {
  1127. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "Unknown phar archive \"%s\"", fname);
  1128. }
  1129. return;
  1130. }
  1131. zname = (char*)zend_get_executed_filename(TSRMLS_C);
  1132. zname_len = strlen(zname);
  1133. if (zname_len > 7 && !memcmp(zname, "phar://", 7) && SUCCESS == phar_split_fname(zname, zname_len, &arch, &arch_len, &entry, &entry_len, 2, 0 TSRMLS_CC)) {
  1134. if (arch_len == fname_len && !memcmp(arch, fname, arch_len)) {
  1135. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar archive \"%s\" cannot be unlinked from within itself", fname);
  1136. efree(arch);
  1137. efree(entry);
  1138. return;
  1139. }
  1140. efree(arch);
  1141. efree(entry);
  1142. }
  1143. if (phar->is_persistent) {
  1144. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar archive \"%s\" is in phar.cache_list, cannot unlinkArchive()", fname);
  1145. return;
  1146. }
  1147. if (phar->refcount) {
  1148. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar archive \"%s\" has open file handles or objects. fclose() all file handles, and unset() all objects prior to calling unlinkArchive()", fname);
  1149. return;
  1150. }
  1151. fname = estrndup(phar->fname, phar->fname_len);
  1152. /* invalidate phar cache */
  1153. PHAR_G(last_phar) = NULL;
  1154. PHAR_G(last_phar_name) = PHAR_G(last_alias) = NULL;
  1155. phar_archive_delref(phar TSRMLS_CC);
  1156. unlink(fname);
  1157. efree(fname);
  1158. RETURN_TRUE;
  1159. }
  1160. /* }}} */
  1161. #if HAVE_SPL
  1162. #define PHAR_ARCHIVE_OBJECT() \
  1163. phar_archive_object *phar_obj = (phar_archive_object*)zend_object_store_get_object(getThis() TSRMLS_CC); \
  1164. if (!phar_obj->arc.archive) { \
  1165. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, \
  1166. "Cannot call method on an uninitialized Phar object"); \
  1167. return; \
  1168. }
  1169. /* {{{ proto void Phar::__destruct()
  1170. * if persistent, remove from the cache
  1171. */
  1172. PHP_METHOD(Phar, __destruct)
  1173. {
  1174. phar_archive_object *phar_obj = (phar_archive_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
  1175. if (phar_obj->arc.archive && phar_obj->arc.archive->is_persistent) {
  1176. zend_hash_del(&PHAR_GLOBALS->phar_persist_map, (const char *) phar_obj->arc.archive, sizeof(phar_obj->arc.archive));
  1177. }
  1178. }
  1179. /* }}} */
  1180. struct _phar_t {
  1181. phar_archive_object *p;
  1182. zend_class_entry *c;
  1183. char *b;
  1184. uint l;
  1185. zval *ret;
  1186. int count;
  1187. php_stream *fp;
  1188. };
  1189. static int phar_build(zend_object_iterator *iter, void *puser TSRMLS_DC) /* {{{ */
  1190. {
  1191. zval **value;
  1192. zend_bool close_fp = 1;
  1193. struct _phar_t *p_obj = (struct _phar_t*) puser;
  1194. uint str_key_len, base_len = p_obj->l, fname_len;
  1195. phar_entry_data *data;
  1196. php_stream *fp;
  1197. size_t contents_len;
  1198. char *fname, *error = NULL, *base = p_obj->b, *opened, *save = NULL, *temp = NULL;
  1199. char *str_key;
  1200. zend_class_entry *ce = p_obj->c;
  1201. phar_archive_object *phar_obj = p_obj->p;
  1202. char *str = "[stream]";
  1203. iter->funcs->get_current_data(iter, &value TSRMLS_CC);
  1204. if (EG(exception)) {
  1205. return ZEND_HASH_APPLY_STOP;
  1206. }
  1207. if (!value) {
  1208. /* failure in get_current_data */
  1209. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "Iterator %v returned no value", ce->name);
  1210. return ZEND_HASH_APPLY_STOP;
  1211. }
  1212. switch (Z_TYPE_PP(value)) {
  1213. case IS_STRING:
  1214. break;
  1215. case IS_RESOURCE:
  1216. php_stream_from_zval_no_verify(fp, value);
  1217. if (!fp) {
  1218. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Iterator %v returned an invalid stream handle", ce->name);
  1219. return ZEND_HASH_APPLY_STOP;
  1220. }
  1221. if (iter->funcs->get_current_key) {
  1222. zval key;
  1223. iter->funcs->get_current_key(iter, &key TSRMLS_CC);
  1224. if (EG(exception)) {
  1225. return ZEND_HASH_APPLY_STOP;
  1226. }
  1227. if (Z_TYPE(key) != IS_STRING) {
  1228. zval_dtor(&key);
  1229. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "Iterator %v returned an invalid key (must return a string)", ce->name);
  1230. return ZEND_HASH_APPLY_STOP;
  1231. }
  1232. str_key_len = Z_STRLEN(key);
  1233. str_key = estrndup(Z_STRVAL(key), str_key_len);
  1234. save = str_key;
  1235. zval_dtor(&key);
  1236. } else {
  1237. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "Iterator %v returned an invalid key (must return a string)", ce->name);
  1238. return ZEND_HASH_APPLY_STOP;
  1239. }
  1240. close_fp = 0;
  1241. opened = (char *) estrndup(str, sizeof("[stream]") - 1);
  1242. goto after_open_fp;
  1243. case IS_OBJECT:
  1244. if (instanceof_function(Z_OBJCE_PP(value), spl_ce_SplFileInfo TSRMLS_CC)) {
  1245. char *test = NULL;
  1246. zval dummy;
  1247. spl_filesystem_object *intern = (spl_filesystem_object*)zend_object_store_get_object(*value TSRMLS_CC);
  1248. if (!base_len) {
  1249. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Iterator %v returns an SplFileInfo object, so base directory must be specified", ce->name);
  1250. return ZEND_HASH_APPLY_STOP;
  1251. }
  1252. switch (intern->type) {
  1253. case SPL_FS_DIR:
  1254. test = spl_filesystem_object_get_path(intern, NULL TSRMLS_CC);
  1255. fname_len = spprintf(&fname, 0, "%s%c%s", test, DEFAULT_SLASH, intern->u.dir.entry.d_name);
  1256. php_stat(fname, fname_len, FS_IS_DIR, &dummy TSRMLS_CC);
  1257. if (Z_BVAL(dummy)) {
  1258. /* ignore directories */
  1259. efree(fname);
  1260. return ZEND_HASH_APPLY_KEEP;
  1261. }
  1262. test = expand_filepath(fname, NULL TSRMLS_CC);
  1263. efree(fname);
  1264. if (test) {
  1265. fname = test;
  1266. fname_len = strlen(fname);
  1267. } else {
  1268. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "Could not resolve file path");
  1269. return ZEND_HASH_APPLY_STOP;
  1270. }
  1271. save = fname;
  1272. goto phar_spl_fileinfo;
  1273. case SPL_FS_INFO:
  1274. case SPL_FS_FILE:
  1275. fname = expand_filepath(intern->file_name, NULL TSRMLS_CC);
  1276. if (!fname) {
  1277. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "Could not resolve file path");
  1278. return ZEND_HASH_APPLY_STOP;
  1279. }
  1280. fname_len = strlen(fname);
  1281. save = fname;
  1282. goto phar_spl_fileinfo;
  1283. }
  1284. }
  1285. /* fall-through */
  1286. default:
  1287. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "Iterator %v returned an invalid value (must return a string)", ce->name);
  1288. return ZEND_HASH_APPLY_STOP;
  1289. }
  1290. fname = Z_STRVAL_PP(value);
  1291. fname_len = Z_STRLEN_PP(value);
  1292. phar_spl_fileinfo:
  1293. if (base_len) {
  1294. temp = expand_filepath(base, NULL TSRMLS_CC);
  1295. if (!temp) {
  1296. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "Could not resolve file path");
  1297. if (save) {
  1298. efree(save);
  1299. }
  1300. return ZEND_HASH_APPLY_STOP;
  1301. }
  1302. base = temp;
  1303. base_len = strlen(base);
  1304. if (strstr(fname, base)) {
  1305. str_key_len = fname_len - base_len;
  1306. if (str_key_len <= 0) {
  1307. if (save) {
  1308. efree(save);
  1309. efree(temp);
  1310. }
  1311. return ZEND_HASH_APPLY_KEEP;
  1312. }
  1313. str_key = fname + base_len;
  1314. if (*str_key == '/' || *str_key == '\\') {
  1315. str_key++;
  1316. str_key_len--;
  1317. }
  1318. } else {
  1319. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "Iterator %v returned a path \"%s\" that is not in the base directory \"%s\"", ce->name, fname, base);
  1320. if (save) {
  1321. efree(save);
  1322. efree(temp);
  1323. }
  1324. return ZEND_HASH_APPLY_STOP;
  1325. }
  1326. } else {
  1327. if (iter->funcs->get_current_key) {
  1328. zval key;
  1329. iter->funcs->get_current_key(iter, &key TSRMLS_CC);
  1330. if (EG(exception)) {
  1331. return ZEND_HASH_APPLY_STOP;
  1332. }
  1333. if (Z_TYPE(key) != IS_STRING) {
  1334. zval_dtor(&key);
  1335. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "Iterator %v returned an invalid key (must return a string)", ce->name);
  1336. return ZEND_HASH_APPLY_STOP;
  1337. }
  1338. str_key_len = Z_STRLEN(key);
  1339. str_key = estrndup(Z_STRVAL(key), str_key_len);
  1340. save = str_key;
  1341. zval_dtor(&key);
  1342. } else {
  1343. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "Iterator %v returned an invalid key (must return a string)", ce->name);
  1344. return ZEND_HASH_APPLY_STOP;
  1345. }
  1346. }
  1347. #if PHP_API_VERSION < 20100412
  1348. if (PG(safe_mode) && (!php_checkuid(fname, NULL, CHECKUID_ALLOW_ONLY_FILE))) {
  1349. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "Iterator %v returned a path \"%s\" that safe mode prevents opening", ce->name, fname);
  1350. if (save) {
  1351. efree(save);
  1352. }
  1353. if (temp) {
  1354. efree(temp);
  1355. }
  1356. return ZEND_HASH_APPLY_STOP;
  1357. }
  1358. #endif
  1359. if (php_check_open_basedir(fname TSRMLS_CC)) {
  1360. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "Iterator %v returned a path \"%s\" that open_basedir prevents opening", ce->name, fname);
  1361. if (save) {
  1362. efree(save);
  1363. }
  1364. if (temp) {
  1365. efree(temp);
  1366. }
  1367. return ZEND_HASH_APPLY_STOP;
  1368. }
  1369. /* try to open source file, then create internal phar file and copy contents */
  1370. fp = php_stream_open_wrapper(fname, "rb", STREAM_MUST_SEEK|0, &opened);
  1371. if (!fp) {
  1372. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "Iterator %v returned a file that could not be opened \"%s\"", ce->name, fname);
  1373. if (save) {
  1374. efree(save);
  1375. }
  1376. if (temp) {
  1377. efree(temp);
  1378. }
  1379. return ZEND_HASH_APPLY_STOP;
  1380. }
  1381. after_open_fp:
  1382. if (str_key_len >= sizeof(".phar")-1 && !memcmp(str_key, ".phar", sizeof(".phar")-1)) {
  1383. /* silently skip any files that would be added to the magic .phar directory */
  1384. if (save) {
  1385. efree(save);
  1386. }
  1387. if (temp) {
  1388. efree(temp);
  1389. }
  1390. if (opened) {
  1391. efree(opened);
  1392. }
  1393. if (close_fp) {
  1394. php_stream_close(fp);
  1395. }
  1396. return ZEND_HASH_APPLY_KEEP;
  1397. }
  1398. if (!(data = phar_get_or_create_entry_data(phar_obj->arc.archive->fname, phar_obj->arc.archive->fname_len, str_key, str_key_len, "w+b", 0, &error, 1 TSRMLS_CC))) {
  1399. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Entry %s cannot be created: %s", str_key, error);
  1400. efree(error);
  1401. if (save) {
  1402. efree(save);
  1403. }
  1404. if (opened) {
  1405. efree(opened);
  1406. }
  1407. if (temp) {
  1408. efree(temp);
  1409. }
  1410. if (close_fp) {
  1411. php_stream_close(fp);
  1412. }
  1413. return ZEND_HASH_APPLY_STOP;
  1414. } else {
  1415. if (error) {
  1416. efree(error);
  1417. }
  1418. /* convert to PHAR_UFP */
  1419. if (data->internal_file->fp_type == PHAR_MOD) {
  1420. php_stream_close(data->internal_file->fp);
  1421. }
  1422. data->internal_file->fp = NULL;
  1423. data->internal_file->fp_type = PHAR_UFP;
  1424. data->internal_file->offset_abs = data->internal_file->offset = php_stream_tell(p_obj->fp);
  1425. data->fp = NULL;
  1426. php_stream_copy_to_stream_ex(fp, p_obj->fp, PHP_STREAM_COPY_ALL, &contents_len);
  1427. data->internal_file->uncompressed_filesize = data->internal_file->compressed_filesize =
  1428. php_stream_tell(p_obj->fp) - data->internal_file->offset;
  1429. }
  1430. if (close_fp) {
  1431. php_stream_close(fp);
  1432. }
  1433. add_assoc_string(p_obj->ret, str_key, opened, 0);
  1434. if (save) {
  1435. efree(save);
  1436. }
  1437. if (temp) {
  1438. efree(temp);
  1439. }
  1440. data->internal_file->compressed_filesize = data->internal_file->uncompressed_filesize = contents_len;
  1441. phar_entry_delref(data TSRMLS_CC);
  1442. return ZEND_HASH_APPLY_KEEP;
  1443. }
  1444. /* }}} */
  1445. /* {{{ proto array Phar::buildFromDirectory(string base_dir[, string regex])
  1446. * Construct a phar archive from an existing directory, recursively.
  1447. * Optional second parameter is a regular expression for filtering directory contents.
  1448. *
  1449. * Return value is an array mapping phar index to actual files added.
  1450. */
  1451. PHP_METHOD(Phar, buildFromDirectory)
  1452. {
  1453. char *dir, *error, *regex = NULL;
  1454. int dir_len, regex_len = 0;
  1455. zend_bool apply_reg = 0;
  1456. zval arg, arg2, *iter, *iteriter, *regexiter = NULL;
  1457. struct _phar_t pass;
  1458. PHAR_ARCHIVE_OBJECT();
  1459. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  1460. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  1461. "Cannot write to archive - write operations restricted by INI setting");
  1462. return;
  1463. }
  1464. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p|s", &dir, &dir_len, &regex, &regex_len) == FAILURE) {
  1465. RETURN_FALSE;
  1466. }
  1467. MAKE_STD_ZVAL(iter);
  1468. if (SUCCESS != object_init_ex(iter, spl_ce_RecursiveDirectoryIterator)) {
  1469. zval_ptr_dtor(&iter);
  1470. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Unable to instantiate directory iterator for %s", phar_obj->arc.archive->fname);
  1471. RETURN_FALSE;
  1472. }
  1473. INIT_PZVAL(&arg);
  1474. ZVAL_STRINGL(&arg, dir, dir_len, 0);
  1475. INIT_PZVAL(&arg2);
  1476. ZVAL_LONG(&arg2, SPL_FILE_DIR_SKIPDOTS|SPL_FILE_DIR_UNIXPATHS);
  1477. zend_call_method_with_2_params(&iter, spl_ce_RecursiveDirectoryIterator,
  1478. &spl_ce_RecursiveDirectoryIterator->constructor, "__construct", NULL, &arg, &arg2);
  1479. if (EG(exception)) {
  1480. zval_ptr_dtor(&iter);
  1481. RETURN_FALSE;
  1482. }
  1483. MAKE_STD_ZVAL(iteriter);
  1484. if (SUCCESS != object_init_ex(iteriter, spl_ce_RecursiveIteratorIterator)) {
  1485. zval_ptr_dtor(&iter);
  1486. zval_ptr_dtor(&iteriter);
  1487. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Unable to instantiate directory iterator for %s", phar_obj->arc.archive->fname);
  1488. RETURN_FALSE;
  1489. }
  1490. zend_call_method_with_1_params(&iteriter, spl_ce_RecursiveIteratorIterator,
  1491. &spl_ce_RecursiveIteratorIterator->constructor, "__construct", NULL, iter);
  1492. if (EG(exception)) {
  1493. zval_ptr_dtor(&iter);
  1494. zval_ptr_dtor(&iteriter);
  1495. RETURN_FALSE;
  1496. }
  1497. zval_ptr_dtor(&iter);
  1498. if (regex_len > 0) {
  1499. apply_reg = 1;
  1500. MAKE_STD_ZVAL(regexiter);
  1501. if (SUCCESS != object_init_ex(regexiter, spl_ce_RegexIterator)) {
  1502. zval_ptr_dtor(&iteriter);
  1503. zval_dtor(regexiter);
  1504. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Unable to instantiate regex iterator for %s", phar_obj->arc.archive->fname);
  1505. RETURN_FALSE;
  1506. }
  1507. INIT_PZVAL(&arg2);
  1508. ZVAL_STRINGL(&arg2, regex, regex_len, 0);
  1509. zend_call_method_with_2_params(&regexiter, spl_ce_RegexIterator,
  1510. &spl_ce_RegexIterator->constructor, "__construct", NULL, iteriter, &arg2);
  1511. }
  1512. array_init(return_value);
  1513. pass.c = apply_reg ? Z_OBJCE_P(regexiter) : Z_OBJCE_P(iteriter);
  1514. pass.p = phar_obj;
  1515. pass.b = dir;
  1516. pass.l = dir_len;
  1517. pass.count = 0;
  1518. pass.ret = return_value;
  1519. pass.fp = php_stream_fopen_tmpfile();
  1520. if (pass.fp == NULL) {
  1521. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" unable to create temporary file", phar_obj->arc.archive->fname);
  1522. return;
  1523. }
  1524. if (phar_obj->arc.archive->is_persistent && FAILURE == phar_copy_on_write(&(phar_obj->arc.archive) TSRMLS_CC)) {
  1525. zval_ptr_dtor(&iteriter);
  1526. if (apply_reg) {
  1527. zval_ptr_dtor(&regexiter);
  1528. }
  1529. php_stream_close(pass.fp);
  1530. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar_obj->arc.archive->fname);
  1531. return;
  1532. }
  1533. if (SUCCESS == spl_iterator_apply((apply_reg ? regexiter : iteriter), (spl_iterator_apply_func_t) phar_build, (void *) &pass TSRMLS_CC)) {
  1534. zval_ptr_dtor(&iteriter);
  1535. if (apply_reg) {
  1536. zval_ptr_dtor(&regexiter);
  1537. }
  1538. phar_obj->arc.archive->ufp = pass.fp;
  1539. phar_flush(phar_obj->arc.archive, 0, 0, 0, &error TSRMLS_CC);
  1540. if (error) {
  1541. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  1542. efree(error);
  1543. }
  1544. } else {
  1545. zval_ptr_dtor(&iteriter);
  1546. if (apply_reg) {
  1547. zval_ptr_dtor(&regexiter);
  1548. }
  1549. php_stream_close(pass.fp);
  1550. }
  1551. }
  1552. /* }}} */
  1553. /* {{{ proto array Phar::buildFromIterator(Iterator iter[, string base_directory])
  1554. * Construct a phar archive from an iterator. The iterator must return a series of strings
  1555. * that are full paths to files that should be added to the phar. The iterator key should
  1556. * be the path that the file will have within the phar archive.
  1557. *
  1558. * If base directory is specified, then the key will be ignored, and instead the portion of
  1559. * the current value minus the base directory will be used
  1560. *
  1561. * Returned is an array mapping phar index to actual file added
  1562. */
  1563. PHP_METHOD(Phar, buildFromIterator)
  1564. {
  1565. zval *obj;
  1566. char *error;
  1567. uint base_len = 0;
  1568. char *base = NULL;
  1569. struct _phar_t pass;
  1570. PHAR_ARCHIVE_OBJECT();
  1571. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  1572. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  1573. "Cannot write out phar archive, phar is read-only");
  1574. return;
  1575. }
  1576. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "O|s", &obj, zend_ce_traversable, &base, &base_len) == FAILURE) {
  1577. RETURN_FALSE;
  1578. }
  1579. if (phar_obj->arc.archive->is_persistent && FAILURE == phar_copy_on_write(&(phar_obj->arc.archive) TSRMLS_CC)) {
  1580. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar_obj->arc.archive->fname);
  1581. return;
  1582. }
  1583. array_init(return_value);
  1584. pass.c = Z_OBJCE_P(obj);
  1585. pass.p = phar_obj;
  1586. pass.b = base;
  1587. pass.l = base_len;
  1588. pass.ret = return_value;
  1589. pass.count = 0;
  1590. pass.fp = php_stream_fopen_tmpfile();
  1591. if (pass.fp == NULL) {
  1592. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\": unable to create temporary file", phar_obj->arc.archive->fname);
  1593. return;
  1594. }
  1595. if (SUCCESS == spl_iterator_apply(obj, (spl_iterator_apply_func_t) phar_build, (void *) &pass TSRMLS_CC)) {
  1596. phar_obj->arc.archive->ufp = pass.fp;
  1597. phar_flush(phar_obj->arc.archive, 0, 0, 0, &error TSRMLS_CC);
  1598. if (error) {
  1599. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  1600. efree(error);
  1601. }
  1602. } else {
  1603. php_stream_close(pass.fp);
  1604. }
  1605. }
  1606. /* }}} */
  1607. /* {{{ proto int Phar::count()
  1608. * Returns the number of entries in the Phar archive
  1609. */
  1610. PHP_METHOD(Phar, count)
  1611. {
  1612. /* mode can be ignored, maximum depth is 1 */
  1613. long mode;
  1614. PHAR_ARCHIVE_OBJECT();
  1615. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|l", &mode) == FAILURE) {
  1616. RETURN_FALSE;
  1617. }
  1618. RETURN_LONG(zend_hash_num_elements(&phar_obj->arc.archive->manifest));
  1619. }
  1620. /* }}} */
  1621. /* {{{ proto bool Phar::isFileFormat(int format)
  1622. * Returns true if the phar archive is based on the tar/zip/phar file format depending
  1623. * on whether Phar::TAR, Phar::ZIP or Phar::PHAR was passed in
  1624. */
  1625. PHP_METHOD(Phar, isFileFormat)
  1626. {
  1627. long type;
  1628. PHAR_ARCHIVE_OBJECT();
  1629. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &type) == FAILURE) {
  1630. RETURN_FALSE;
  1631. }
  1632. switch (type) {
  1633. case PHAR_FORMAT_TAR:
  1634. RETURN_BOOL(phar_obj->arc.archive->is_tar);
  1635. case PHAR_FORMAT_ZIP:
  1636. RETURN_BOOL(phar_obj->arc.archive->is_zip);
  1637. case PHAR_FORMAT_PHAR:
  1638. RETURN_BOOL(!phar_obj->arc.archive->is_tar && !phar_obj->arc.archive->is_zip);
  1639. default:
  1640. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "Unknown file format specified");
  1641. }
  1642. }
  1643. /* }}} */
  1644. static int phar_copy_file_contents(phar_entry_info *entry, php_stream *fp TSRMLS_DC) /* {{{ */
  1645. {
  1646. char *error;
  1647. off_t offset;
  1648. phar_entry_info *link;
  1649. if (FAILURE == phar_open_entry_fp(entry, &error, 1 TSRMLS_CC)) {
  1650. if (error) {
  1651. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  1652. "Cannot convert phar archive \"%s\", unable to open entry \"%s\" contents: %s", entry->phar->fname, entry->filename, error);
  1653. efree(error);
  1654. } else {
  1655. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  1656. "Cannot convert phar archive \"%s\", unable to open entry \"%s\" contents", entry->phar->fname, entry->filename);
  1657. }
  1658. return FAILURE;
  1659. }
  1660. /* copy old contents in entirety */
  1661. phar_seek_efp(entry, 0, SEEK_SET, 0, 1 TSRMLS_CC);
  1662. offset = php_stream_tell(fp);
  1663. link = phar_get_link_source(entry TSRMLS_CC);
  1664. if (!link) {
  1665. link = entry;
  1666. }
  1667. if (SUCCESS != php_stream_copy_to_stream_ex(phar_get_efp(link, 0 TSRMLS_CC), fp, link->uncompressed_filesize, NULL)) {
  1668. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  1669. "Cannot convert phar archive \"%s\", unable to copy entry \"%s\" contents", entry->phar->fname, entry->filename);
  1670. return FAILURE;
  1671. }
  1672. if (entry->fp_type == PHAR_MOD) {
  1673. /* save for potential restore on error */
  1674. entry->cfp = entry->fp;
  1675. entry->fp = NULL;
  1676. }
  1677. /* set new location of file contents */
  1678. entry->fp_type = PHAR_FP;
  1679. entry->offset = offset;
  1680. return SUCCESS;
  1681. }
  1682. /* }}} */
  1683. static zval *phar_rename_archive(phar_archive_data **sphar, char *ext, zend_bool compress TSRMLS_DC) /* {{{ */
  1684. {
  1685. const char *oldname = NULL;
  1686. phar_archive_data *phar = *sphar;
  1687. char *oldpath = NULL;
  1688. char *basename = NULL, *basepath = NULL;
  1689. char *newname = NULL, *newpath = NULL;
  1690. zval *ret, arg1;
  1691. zend_class_entry *ce;
  1692. char *error;
  1693. const char *pcr_error;
  1694. int ext_len = ext ? strlen(ext) : 0;
  1695. int oldname_len;
  1696. phar_archive_data **pphar = NULL;
  1697. php_stream_statbuf ssb;
  1698. if (!ext) {
  1699. if (phar->is_zip) {
  1700. if (phar->is_data) {
  1701. ext = "zip";
  1702. } else {
  1703. ext = "phar.zip";
  1704. }
  1705. } else if (phar->is_tar) {
  1706. switch (phar->flags) {
  1707. case PHAR_FILE_COMPRESSED_GZ:
  1708. if (phar->is_data) {
  1709. ext = "tar.gz";
  1710. } else {
  1711. ext = "phar.tar.gz";
  1712. }
  1713. break;
  1714. case PHAR_FILE_COMPRESSED_BZ2:
  1715. if (phar->is_data) {
  1716. ext = "tar.bz2";
  1717. } else {
  1718. ext = "phar.tar.bz2";
  1719. }
  1720. break;
  1721. default:
  1722. if (phar->is_data) {
  1723. ext = "tar";
  1724. } else {
  1725. ext = "phar.tar";
  1726. }
  1727. }
  1728. } else {
  1729. switch (phar->flags) {
  1730. case PHAR_FILE_COMPRESSED_GZ:
  1731. ext = "phar.gz";
  1732. break;
  1733. case PHAR_FILE_COMPRESSED_BZ2:
  1734. ext = "phar.bz2";
  1735. break;
  1736. default:
  1737. ext = "phar";
  1738. }
  1739. }
  1740. } else if (phar_path_check(&ext, &ext_len, &pcr_error) > pcr_is_ok) {
  1741. if (phar->is_data) {
  1742. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "data phar converted from \"%s\" has invalid extension %s", phar->fname, ext);
  1743. } else {
  1744. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "phar converted from \"%s\" has invalid extension %s", phar->fname, ext);
  1745. }
  1746. return NULL;
  1747. }
  1748. if (ext[0] == '.') {
  1749. ++ext;
  1750. }
  1751. oldpath = estrndup(phar->fname, phar->fname_len);
  1752. oldname = zend_memrchr(phar->fname, '/', phar->fname_len);
  1753. ++oldname;
  1754. oldname_len = strlen(oldname);
  1755. basename = estrndup(oldname, oldname_len);
  1756. spprintf(&newname, 0, "%s.%s", strtok(basename, "."), ext);
  1757. efree(basename);
  1758. basepath = estrndup(oldpath, (strlen(oldpath) - oldname_len));
  1759. phar->fname_len = spprintf(&newpath, 0, "%s%s", basepath, newname);
  1760. phar->fname = newpath;
  1761. phar->ext = newpath + phar->fname_len - strlen(ext) - 1;
  1762. efree(basepath);
  1763. efree(newname);
  1764. if (PHAR_G(manifest_cached) && SUCCESS == zend_hash_find(&cached_phars, newpath, phar->fname_len, (void **) &pphar)) {
  1765. efree(oldpath);
  1766. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Unable to add newly converted phar \"%s\" to the list of phars, new phar name is in phar.cache_list", phar->fname);
  1767. return NULL;
  1768. }
  1769. if (SUCCESS == zend_hash_find(&(PHAR_GLOBALS->phar_fname_map), newpath, phar->fname_len, (void **) &pphar)) {
  1770. if ((*pphar)->fname_len == phar->fname_len && !memcmp((*pphar)->fname, phar->fname, phar->fname_len)) {
  1771. if (!zend_hash_num_elements(&phar->manifest)) {
  1772. (*pphar)->is_tar = phar->is_tar;
  1773. (*pphar)->is_zip = phar->is_zip;
  1774. (*pphar)->is_data = phar->is_data;
  1775. (*pphar)->flags = phar->flags;
  1776. (*pphar)->fp = phar->fp;
  1777. phar->fp = NULL;
  1778. phar_destroy_phar_data(phar TSRMLS_CC);
  1779. *sphar = NULL;
  1780. phar = *pphar;
  1781. *sphar = NULL;
  1782. phar->refcount++;
  1783. newpath = oldpath;
  1784. goto its_ok;
  1785. }
  1786. }
  1787. efree(oldpath);
  1788. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Unable to add newly converted phar \"%s\" to the list of phars, a phar with that name already exists", phar->fname);
  1789. return NULL;
  1790. }
  1791. its_ok:
  1792. if (SUCCESS == php_stream_stat_path(newpath, &ssb)) {
  1793. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "phar \"%s\" exists and must be unlinked prior to conversion", newpath);
  1794. efree(oldpath);
  1795. return NULL;
  1796. }
  1797. if (!phar->is_data) {
  1798. if (SUCCESS != phar_detect_phar_fname_ext(newpath, phar->fname_len, (const char **) &(phar->ext), &(phar->ext_len), 1, 1, 1 TSRMLS_CC)) {
  1799. efree(oldpath);
  1800. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "phar \"%s\" has invalid extension %s", phar->fname, ext);
  1801. return NULL;
  1802. }
  1803. if (phar->alias) {
  1804. if (phar->is_temporary_alias) {
  1805. phar->alias = NULL;
  1806. phar->alias_len = 0;
  1807. } else {
  1808. phar->alias = estrndup(newpath, strlen(newpath));
  1809. phar->alias_len = strlen(newpath);
  1810. phar->is_temporary_alias = 1;
  1811. zend_hash_update(&(PHAR_GLOBALS->phar_alias_map), newpath, phar->fname_len, (void*)&phar, sizeof(phar_archive_data*), NULL);
  1812. }
  1813. }
  1814. } else {
  1815. if (SUCCESS != phar_detect_phar_fname_ext(newpath, phar->fname_len, (const char **) &(phar->ext), &(phar->ext_len), 0, 1, 1 TSRMLS_CC)) {
  1816. efree(oldpath);
  1817. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "data phar \"%s\" has invalid extension %s", phar->fname, ext);
  1818. return NULL;
  1819. }
  1820. phar->alias = NULL;
  1821. phar->alias_len = 0;
  1822. }
  1823. if ((!pphar || phar == *pphar) && SUCCESS != zend_hash_update(&(PHAR_GLOBALS->phar_fname_map), newpath, phar->fname_len, (void*)&phar, sizeof(phar_archive_data*), NULL)) {
  1824. efree(oldpath);
  1825. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Unable to add newly converted phar \"%s\" to the list of phars", phar->fname);
  1826. return NULL;
  1827. }
  1828. phar_flush(phar, 0, 0, 1, &error TSRMLS_CC);
  1829. if (error) {
  1830. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "%s", error);
  1831. efree(error);
  1832. efree(oldpath);
  1833. return NULL;
  1834. }
  1835. efree(oldpath);
  1836. if (phar->is_data) {
  1837. ce = phar_ce_data;
  1838. } else {
  1839. ce = phar_ce_archive;
  1840. }
  1841. MAKE_STD_ZVAL(ret);
  1842. if (SUCCESS != object_init_ex(ret, ce)) {
  1843. zval_dtor(ret);
  1844. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Unable to instantiate phar object when converting archive \"%s\"", phar->fname);
  1845. return NULL;
  1846. }
  1847. INIT_PZVAL(&arg1);
  1848. ZVAL_STRINGL(&arg1, phar->fname, phar->fname_len, 0);
  1849. zend_call_method_with_1_params(&ret, ce, &ce->constructor, "__construct", NULL, &arg1);
  1850. return ret;
  1851. }
  1852. /* }}} */
  1853. static zval *phar_convert_to_other(phar_archive_data *source, int convert, char *ext, php_uint32 flags TSRMLS_DC) /* {{{ */
  1854. {
  1855. phar_archive_data *phar;
  1856. phar_entry_info *entry, newentry;
  1857. zval *ret;
  1858. /* invalidate phar cache */
  1859. PHAR_G(last_phar) = NULL;
  1860. PHAR_G(last_phar_name) = PHAR_G(last_alias) = NULL;
  1861. phar = (phar_archive_data *) ecalloc(1, sizeof(phar_archive_data));
  1862. /* set whole-archive compression and type from parameter */
  1863. phar->flags = flags;
  1864. phar->is_data = source->is_data;
  1865. switch (convert) {
  1866. case PHAR_FORMAT_TAR:
  1867. phar->is_tar = 1;
  1868. break;
  1869. case PHAR_FORMAT_ZIP:
  1870. phar->is_zip = 1;
  1871. break;
  1872. default:
  1873. phar->is_data = 0;
  1874. break;
  1875. }
  1876. zend_hash_init(&(phar->manifest), sizeof(phar_entry_info),
  1877. zend_get_hash_value, destroy_phar_manifest_entry, 0);
  1878. zend_hash_init(&phar->mounted_dirs, sizeof(char *),
  1879. zend_get_hash_value, NULL, 0);
  1880. zend_hash_init(&phar->virtual_dirs, sizeof(char *),
  1881. zend_get_hash_value, NULL, 0);
  1882. phar->fp = php_stream_fopen_tmpfile();
  1883. if (phar->fp == NULL) {
  1884. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "unable to create temporary file");
  1885. return NULL;
  1886. }
  1887. phar->fname = source->fname;
  1888. phar->fname_len = source->fname_len;
  1889. phar->is_temporary_alias = source->is_temporary_alias;
  1890. phar->alias = source->alias;
  1891. if (source->metadata) {
  1892. zval *t;
  1893. t = source->metadata;
  1894. ALLOC_ZVAL(phar->metadata);
  1895. *phar->metadata = *t;
  1896. zval_copy_ctor(phar->metadata);
  1897. Z_SET_REFCOUNT_P(phar->metadata, 1);
  1898. phar->metadata_len = 0;
  1899. }
  1900. /* first copy each file's uncompressed contents to a temporary file and set per-file flags */
  1901. for (zend_hash_internal_pointer_reset(&source->manifest); SUCCESS == zend_hash_has_more_elements(&source->manifest); zend_hash_move_forward(&source->manifest)) {
  1902. if (FAILURE == zend_hash_get_current_data(&source->manifest, (void **) &entry)) {
  1903. zend_hash_destroy(&(phar->manifest));
  1904. php_stream_close(phar->fp);
  1905. efree(phar);
  1906. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  1907. "Cannot convert phar archive \"%s\"", source->fname);
  1908. return NULL;
  1909. }
  1910. newentry = *entry;
  1911. if (newentry.link) {
  1912. newentry.link = estrdup(newentry.link);
  1913. goto no_copy;
  1914. }
  1915. if (newentry.tmp) {
  1916. newentry.tmp = estrdup(newentry.tmp);
  1917. goto no_copy;
  1918. }
  1919. newentry.metadata_str.c = 0;
  1920. if (FAILURE == phar_copy_file_contents(&newentry, phar->fp TSRMLS_CC)) {
  1921. zend_hash_destroy(&(phar->manifest));
  1922. php_stream_close(phar->fp);
  1923. efree(phar);
  1924. /* exception already thrown */
  1925. return NULL;
  1926. }
  1927. no_copy:
  1928. newentry.filename = estrndup(newentry.filename, newentry.filename_len);
  1929. if (newentry.metadata) {
  1930. zval *t;
  1931. t = newentry.metadata;
  1932. ALLOC_ZVAL(newentry.metadata);
  1933. *newentry.metadata = *t;
  1934. zval_copy_ctor(newentry.metadata);
  1935. Z_SET_REFCOUNT_P(newentry.metadata, 1);
  1936. newentry.metadata_str.c = NULL;
  1937. newentry.metadata_str.len = 0;
  1938. }
  1939. newentry.is_zip = phar->is_zip;
  1940. newentry.is_tar = phar->is_tar;
  1941. if (newentry.is_tar) {
  1942. newentry.tar_type = (entry->is_dir ? TAR_DIR : TAR_FILE);
  1943. }
  1944. newentry.is_modified = 1;
  1945. newentry.phar = phar;
  1946. newentry.old_flags = newentry.flags & ~PHAR_ENT_COMPRESSION_MASK; /* remove compression from old_flags */
  1947. phar_set_inode(&newentry TSRMLS_CC);
  1948. zend_hash_add(&(phar->manifest), newentry.filename, newentry.filename_len, (void*)&newentry, sizeof(phar_entry_info), NULL);
  1949. phar_add_virtual_dirs(phar, newentry.filename, newentry.filename_len TSRMLS_CC);
  1950. }
  1951. if ((ret = phar_rename_archive(&phar, ext, 0 TSRMLS_CC))) {
  1952. return ret;
  1953. } else {
  1954. if(phar != NULL) {
  1955. zend_hash_destroy(&(phar->manifest));
  1956. zend_hash_destroy(&(phar->mounted_dirs));
  1957. zend_hash_destroy(&(phar->virtual_dirs));
  1958. if (phar->fp) {
  1959. php_stream_close(phar->fp);
  1960. }
  1961. efree(phar->fname);
  1962. efree(phar);
  1963. }
  1964. return NULL;
  1965. }
  1966. }
  1967. /* }}} */
  1968. /* {{{ proto object Phar::convertToExecutable([int format[, int compression [, string file_ext]]])
  1969. * Convert a phar.tar or phar.zip archive to the phar file format. The
  1970. * optional parameter allows the user to determine the new
  1971. * filename extension (default is phar).
  1972. */
  1973. PHP_METHOD(Phar, convertToExecutable)
  1974. {
  1975. char *ext = NULL;
  1976. int is_data, ext_len = 0;
  1977. php_uint32 flags;
  1978. zval *ret;
  1979. /* a number that is not 0, 1 or 2 (Which is also Greg's birthday, so there) */
  1980. long format = 9021976, method = 9021976;
  1981. PHAR_ARCHIVE_OBJECT();
  1982. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|lls", &format, &method, &ext, &ext_len) == FAILURE) {
  1983. return;
  1984. }
  1985. if (PHAR_G(readonly)) {
  1986. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  1987. "Cannot write out executable phar archive, phar is read-only");
  1988. return;
  1989. }
  1990. switch (format) {
  1991. case 9021976:
  1992. case PHAR_FORMAT_SAME: /* null is converted to 0 */
  1993. /* by default, use the existing format */
  1994. if (phar_obj->arc.archive->is_tar) {
  1995. format = PHAR_FORMAT_TAR;
  1996. } else if (phar_obj->arc.archive->is_zip) {
  1997. format = PHAR_FORMAT_ZIP;
  1998. } else {
  1999. format = PHAR_FORMAT_PHAR;
  2000. }
  2001. break;
  2002. case PHAR_FORMAT_PHAR:
  2003. case PHAR_FORMAT_TAR:
  2004. case PHAR_FORMAT_ZIP:
  2005. break;
  2006. default:
  2007. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2008. "Unknown file format specified, please pass one of Phar::PHAR, Phar::TAR or Phar::ZIP");
  2009. return;
  2010. }
  2011. switch (method) {
  2012. case 9021976:
  2013. flags = phar_obj->arc.archive->flags & PHAR_FILE_COMPRESSION_MASK;
  2014. break;
  2015. case 0:
  2016. flags = PHAR_FILE_COMPRESSED_NONE;
  2017. break;
  2018. case PHAR_ENT_COMPRESSED_GZ:
  2019. if (format == PHAR_FORMAT_ZIP) {
  2020. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2021. "Cannot compress entire archive with gzip, zip archives do not support whole-archive compression");
  2022. return;
  2023. }
  2024. if (!PHAR_G(has_zlib)) {
  2025. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2026. "Cannot compress entire archive with gzip, enable ext/zlib in php.ini");
  2027. return;
  2028. }
  2029. flags = PHAR_FILE_COMPRESSED_GZ;
  2030. break;
  2031. case PHAR_ENT_COMPRESSED_BZ2:
  2032. if (format == PHAR_FORMAT_ZIP) {
  2033. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2034. "Cannot compress entire archive with bz2, zip archives do not support whole-archive compression");
  2035. return;
  2036. }
  2037. if (!PHAR_G(has_bz2)) {
  2038. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2039. "Cannot compress entire archive with bz2, enable ext/bz2 in php.ini");
  2040. return;
  2041. }
  2042. flags = PHAR_FILE_COMPRESSED_BZ2;
  2043. break;
  2044. default:
  2045. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2046. "Unknown compression specified, please pass one of Phar::GZ or Phar::BZ2");
  2047. return;
  2048. }
  2049. is_data = phar_obj->arc.archive->is_data;
  2050. phar_obj->arc.archive->is_data = 0;
  2051. ret = phar_convert_to_other(phar_obj->arc.archive, format, ext, flags TSRMLS_CC);
  2052. phar_obj->arc.archive->is_data = is_data;
  2053. if (ret) {
  2054. RETURN_ZVAL(ret, 1, 1);
  2055. } else {
  2056. RETURN_NULL();
  2057. }
  2058. }
  2059. /* }}} */
  2060. /* {{{ proto object Phar::convertToData([int format[, int compression [, string file_ext]]])
  2061. * Convert an archive to a non-executable .tar or .zip.
  2062. * The optional parameter allows the user to determine the new
  2063. * filename extension (default is .zip or .tar).
  2064. */
  2065. PHP_METHOD(Phar, convertToData)
  2066. {
  2067. char *ext = NULL;
  2068. int is_data, ext_len = 0;
  2069. php_uint32 flags;
  2070. zval *ret;
  2071. /* a number that is not 0, 1 or 2 (Which is also Greg's birthday so there) */
  2072. long format = 9021976, method = 9021976;
  2073. PHAR_ARCHIVE_OBJECT();
  2074. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|lls", &format, &method, &ext, &ext_len) == FAILURE) {
  2075. return;
  2076. }
  2077. switch (format) {
  2078. case 9021976:
  2079. case PHAR_FORMAT_SAME: /* null is converted to 0 */
  2080. /* by default, use the existing format */
  2081. if (phar_obj->arc.archive->is_tar) {
  2082. format = PHAR_FORMAT_TAR;
  2083. } else if (phar_obj->arc.archive->is_zip) {
  2084. format = PHAR_FORMAT_ZIP;
  2085. } else {
  2086. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2087. "Cannot write out data phar archive, use Phar::TAR or Phar::ZIP");
  2088. return;
  2089. }
  2090. break;
  2091. case PHAR_FORMAT_PHAR:
  2092. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2093. "Cannot write out data phar archive, use Phar::TAR or Phar::ZIP");
  2094. return;
  2095. case PHAR_FORMAT_TAR:
  2096. case PHAR_FORMAT_ZIP:
  2097. break;
  2098. default:
  2099. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2100. "Unknown file format specified, please pass one of Phar::TAR or Phar::ZIP");
  2101. return;
  2102. }
  2103. switch (method) {
  2104. case 9021976:
  2105. flags = phar_obj->arc.archive->flags & PHAR_FILE_COMPRESSION_MASK;
  2106. break;
  2107. case 0:
  2108. flags = PHAR_FILE_COMPRESSED_NONE;
  2109. break;
  2110. case PHAR_ENT_COMPRESSED_GZ:
  2111. if (format == PHAR_FORMAT_ZIP) {
  2112. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2113. "Cannot compress entire archive with gzip, zip archives do not support whole-archive compression");
  2114. return;
  2115. }
  2116. if (!PHAR_G(has_zlib)) {
  2117. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2118. "Cannot compress entire archive with gzip, enable ext/zlib in php.ini");
  2119. return;
  2120. }
  2121. flags = PHAR_FILE_COMPRESSED_GZ;
  2122. break;
  2123. case PHAR_ENT_COMPRESSED_BZ2:
  2124. if (format == PHAR_FORMAT_ZIP) {
  2125. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2126. "Cannot compress entire archive with bz2, zip archives do not support whole-archive compression");
  2127. return;
  2128. }
  2129. if (!PHAR_G(has_bz2)) {
  2130. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2131. "Cannot compress entire archive with bz2, enable ext/bz2 in php.ini");
  2132. return;
  2133. }
  2134. flags = PHAR_FILE_COMPRESSED_BZ2;
  2135. break;
  2136. default:
  2137. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2138. "Unknown compression specified, please pass one of Phar::GZ or Phar::BZ2");
  2139. return;
  2140. }
  2141. is_data = phar_obj->arc.archive->is_data;
  2142. phar_obj->arc.archive->is_data = 1;
  2143. ret = phar_convert_to_other(phar_obj->arc.archive, format, ext, flags TSRMLS_CC);
  2144. phar_obj->arc.archive->is_data = is_data;
  2145. if (ret) {
  2146. RETURN_ZVAL(ret, 1, 1);
  2147. } else {
  2148. RETURN_NULL();
  2149. }
  2150. }
  2151. /* }}} */
  2152. /* {{{ proto int|false Phar::isCompressed()
  2153. * Returns Phar::GZ or PHAR::BZ2 if the entire archive is compressed
  2154. * (.tar.gz/tar.bz2 and so on), or FALSE otherwise.
  2155. */
  2156. PHP_METHOD(Phar, isCompressed)
  2157. {
  2158. PHAR_ARCHIVE_OBJECT();
  2159. if (zend_parse_parameters_none() == FAILURE) {
  2160. return;
  2161. }
  2162. if (phar_obj->arc.archive->flags & PHAR_FILE_COMPRESSED_GZ) {
  2163. RETURN_LONG(PHAR_ENT_COMPRESSED_GZ);
  2164. }
  2165. if (phar_obj->arc.archive->flags & PHAR_FILE_COMPRESSED_BZ2) {
  2166. RETURN_LONG(PHAR_ENT_COMPRESSED_BZ2);
  2167. }
  2168. RETURN_FALSE;
  2169. }
  2170. /* }}} */
  2171. /* {{{ proto bool Phar::isWritable()
  2172. * Returns true if phar.readonly=0 or phar is a PharData AND the actual file is writable.
  2173. */
  2174. PHP_METHOD(Phar, isWritable)
  2175. {
  2176. php_stream_statbuf ssb;
  2177. PHAR_ARCHIVE_OBJECT();
  2178. if (zend_parse_parameters_none() == FAILURE) {
  2179. return;
  2180. }
  2181. if (!phar_obj->arc.archive->is_writeable) {
  2182. RETURN_FALSE;
  2183. }
  2184. if (SUCCESS != php_stream_stat_path(phar_obj->arc.archive->fname, &ssb)) {
  2185. if (phar_obj->arc.archive->is_brandnew) {
  2186. /* assume it works if the file doesn't exist yet */
  2187. RETURN_TRUE;
  2188. }
  2189. RETURN_FALSE;
  2190. }
  2191. RETURN_BOOL((ssb.sb.st_mode & (S_IWOTH | S_IWGRP | S_IWUSR)) != 0);
  2192. }
  2193. /* }}} */
  2194. /* {{{ proto bool Phar::delete(string entry)
  2195. * Deletes a named file within the archive.
  2196. */
  2197. PHP_METHOD(Phar, delete)
  2198. {
  2199. char *fname;
  2200. int fname_len;
  2201. char *error;
  2202. phar_entry_info *entry;
  2203. PHAR_ARCHIVE_OBJECT();
  2204. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  2205. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2206. "Cannot write out phar archive, phar is read-only");
  2207. return;
  2208. }
  2209. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p", &fname, &fname_len) == FAILURE) {
  2210. RETURN_FALSE;
  2211. }
  2212. if (phar_obj->arc.archive->is_persistent && FAILURE == phar_copy_on_write(&(phar_obj->arc.archive) TSRMLS_CC)) {
  2213. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar_obj->arc.archive->fname);
  2214. return;
  2215. }
  2216. if (zend_hash_exists(&phar_obj->arc.archive->manifest, fname, (uint) fname_len)) {
  2217. if (SUCCESS == zend_hash_find(&phar_obj->arc.archive->manifest, fname, (uint) fname_len, (void**)&entry)) {
  2218. if (entry->is_deleted) {
  2219. /* entry is deleted, but has not been flushed to disk yet */
  2220. RETURN_TRUE;
  2221. } else {
  2222. entry->is_deleted = 1;
  2223. entry->is_modified = 1;
  2224. phar_obj->arc.archive->is_modified = 1;
  2225. }
  2226. }
  2227. } else {
  2228. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Entry %s does not exist and cannot be deleted", fname);
  2229. RETURN_FALSE;
  2230. }
  2231. phar_flush(phar_obj->arc.archive, NULL, 0, 0, &error TSRMLS_CC);
  2232. if (error) {
  2233. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  2234. efree(error);
  2235. }
  2236. RETURN_TRUE;
  2237. }
  2238. /* }}} */
  2239. /* {{{ proto int Phar::getAlias()
  2240. * Returns the alias for the Phar or NULL.
  2241. */
  2242. PHP_METHOD(Phar, getAlias)
  2243. {
  2244. PHAR_ARCHIVE_OBJECT();
  2245. if (zend_parse_parameters_none() == FAILURE) {
  2246. return;
  2247. }
  2248. if (phar_obj->arc.archive->alias && phar_obj->arc.archive->alias != phar_obj->arc.archive->fname) {
  2249. RETURN_STRINGL(phar_obj->arc.archive->alias, phar_obj->arc.archive->alias_len, 1);
  2250. }
  2251. }
  2252. /* }}} */
  2253. /* {{{ proto int Phar::getPath()
  2254. * Returns the real path to the phar archive on disk
  2255. */
  2256. PHP_METHOD(Phar, getPath)
  2257. {
  2258. PHAR_ARCHIVE_OBJECT();
  2259. if (zend_parse_parameters_none() == FAILURE) {
  2260. return;
  2261. }
  2262. RETURN_STRINGL(phar_obj->arc.archive->fname, phar_obj->arc.archive->fname_len, 1);
  2263. }
  2264. /* }}} */
  2265. /* {{{ proto bool Phar::setAlias(string alias)
  2266. * Sets the alias for a Phar archive. The default value is the full path
  2267. * to the archive.
  2268. */
  2269. PHP_METHOD(Phar, setAlias)
  2270. {
  2271. char *alias, *error, *oldalias;
  2272. phar_archive_data **fd_ptr;
  2273. int alias_len, oldalias_len, old_temp, readd = 0;
  2274. PHAR_ARCHIVE_OBJECT();
  2275. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  2276. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2277. "Cannot write out phar archive, phar is read-only");
  2278. RETURN_FALSE;
  2279. }
  2280. /* invalidate phar cache */
  2281. PHAR_G(last_phar) = NULL;
  2282. PHAR_G(last_phar_name) = PHAR_G(last_alias) = NULL;
  2283. if (phar_obj->arc.archive->is_data) {
  2284. if (phar_obj->arc.archive->is_tar) {
  2285. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2286. "A Phar alias cannot be set in a plain tar archive");
  2287. } else {
  2288. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2289. "A Phar alias cannot be set in a plain zip archive");
  2290. }
  2291. RETURN_FALSE;
  2292. }
  2293. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s", &alias, &alias_len) == SUCCESS) {
  2294. if (alias_len == phar_obj->arc.archive->alias_len && memcmp(phar_obj->arc.archive->alias, alias, alias_len) == 0) {
  2295. RETURN_TRUE;
  2296. }
  2297. if (alias_len && SUCCESS == zend_hash_find(&(PHAR_GLOBALS->phar_alias_map), alias, alias_len, (void**)&fd_ptr)) {
  2298. spprintf(&error, 0, "alias \"%s\" is already used for archive \"%s\" and cannot be used for other archives", alias, (*fd_ptr)->fname);
  2299. if (SUCCESS == phar_free_alias(*fd_ptr, alias, alias_len TSRMLS_CC)) {
  2300. efree(error);
  2301. goto valid_alias;
  2302. }
  2303. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  2304. efree(error);
  2305. RETURN_FALSE;
  2306. }
  2307. if (!phar_validate_alias(alias, alias_len)) {
  2308. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2309. "Invalid alias \"%s\" specified for phar \"%s\"", alias, phar_obj->arc.archive->fname);
  2310. RETURN_FALSE;
  2311. }
  2312. valid_alias:
  2313. if (phar_obj->arc.archive->is_persistent && FAILURE == phar_copy_on_write(&(phar_obj->arc.archive) TSRMLS_CC)) {
  2314. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar_obj->arc.archive->fname);
  2315. return;
  2316. }
  2317. if (phar_obj->arc.archive->alias_len && SUCCESS == zend_hash_find(&(PHAR_GLOBALS->phar_alias_map), phar_obj->arc.archive->alias, phar_obj->arc.archive->alias_len, (void**)&fd_ptr)) {
  2318. zend_hash_del(&(PHAR_GLOBALS->phar_alias_map), phar_obj->arc.archive->alias, phar_obj->arc.archive->alias_len);
  2319. readd = 1;
  2320. }
  2321. oldalias = phar_obj->arc.archive->alias;
  2322. oldalias_len = phar_obj->arc.archive->alias_len;
  2323. old_temp = phar_obj->arc.archive->is_temporary_alias;
  2324. if (alias_len) {
  2325. phar_obj->arc.archive->alias = estrndup(alias, alias_len);
  2326. } else {
  2327. phar_obj->arc.archive->alias = NULL;
  2328. }
  2329. phar_obj->arc.archive->alias_len = alias_len;
  2330. phar_obj->arc.archive->is_temporary_alias = 0;
  2331. phar_flush(phar_obj->arc.archive, NULL, 0, 0, &error TSRMLS_CC);
  2332. if (error) {
  2333. phar_obj->arc.archive->alias = oldalias;
  2334. phar_obj->arc.archive->alias_len = oldalias_len;
  2335. phar_obj->arc.archive->is_temporary_alias = old_temp;
  2336. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  2337. if (readd) {
  2338. zend_hash_add(&(PHAR_GLOBALS->phar_alias_map), oldalias, oldalias_len, (void*)&(phar_obj->arc.archive), sizeof(phar_archive_data*), NULL);
  2339. }
  2340. efree(error);
  2341. RETURN_FALSE;
  2342. }
  2343. zend_hash_add(&(PHAR_GLOBALS->phar_alias_map), alias, alias_len, (void*)&(phar_obj->arc.archive), sizeof(phar_archive_data*), NULL);
  2344. if (oldalias) {
  2345. efree(oldalias);
  2346. }
  2347. RETURN_TRUE;
  2348. }
  2349. RETURN_FALSE;
  2350. }
  2351. /* }}} */
  2352. /* {{{ proto string Phar::getVersion()
  2353. * Return version info of Phar archive
  2354. */
  2355. PHP_METHOD(Phar, getVersion)
  2356. {
  2357. PHAR_ARCHIVE_OBJECT();
  2358. if (zend_parse_parameters_none() == FAILURE) {
  2359. return;
  2360. }
  2361. RETURN_STRING(phar_obj->arc.archive->version, 1);
  2362. }
  2363. /* }}} */
  2364. /* {{{ proto void Phar::startBuffering()
  2365. * Do not flush a writeable phar (save its contents) until explicitly requested
  2366. */
  2367. PHP_METHOD(Phar, startBuffering)
  2368. {
  2369. PHAR_ARCHIVE_OBJECT();
  2370. if (zend_parse_parameters_none() == FAILURE) {
  2371. return;
  2372. }
  2373. phar_obj->arc.archive->donotflush = 1;
  2374. }
  2375. /* }}} */
  2376. /* {{{ proto bool Phar::isBuffering()
  2377. * Returns whether write operations are flushing to disk immediately.
  2378. */
  2379. PHP_METHOD(Phar, isBuffering)
  2380. {
  2381. PHAR_ARCHIVE_OBJECT();
  2382. if (zend_parse_parameters_none() == FAILURE) {
  2383. return;
  2384. }
  2385. RETURN_BOOL(phar_obj->arc.archive->donotflush);
  2386. }
  2387. /* }}} */
  2388. /* {{{ proto bool Phar::stopBuffering()
  2389. * Saves the contents of a modified archive to disk.
  2390. */
  2391. PHP_METHOD(Phar, stopBuffering)
  2392. {
  2393. char *error;
  2394. PHAR_ARCHIVE_OBJECT();
  2395. if (zend_parse_parameters_none() == FAILURE) {
  2396. return;
  2397. }
  2398. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  2399. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2400. "Cannot write out phar archive, phar is read-only");
  2401. return;
  2402. }
  2403. phar_obj->arc.archive->donotflush = 0;
  2404. phar_flush(phar_obj->arc.archive, 0, 0, 0, &error TSRMLS_CC);
  2405. if (error) {
  2406. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  2407. efree(error);
  2408. }
  2409. }
  2410. /* }}} */
  2411. /* {{{ proto bool Phar::setStub(string|stream stub [, int len])
  2412. * Change the stub in a phar, phar.tar or phar.zip archive to something other
  2413. * than the default. The stub *must* end with a call to __HALT_COMPILER().
  2414. */
  2415. PHP_METHOD(Phar, setStub)
  2416. {
  2417. zval *zstub;
  2418. char *stub, *error;
  2419. int stub_len;
  2420. long len = -1;
  2421. php_stream *stream;
  2422. PHAR_ARCHIVE_OBJECT();
  2423. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  2424. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2425. "Cannot change stub, phar is read-only");
  2426. return;
  2427. }
  2428. if (phar_obj->arc.archive->is_data) {
  2429. if (phar_obj->arc.archive->is_tar) {
  2430. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2431. "A Phar stub cannot be set in a plain tar archive");
  2432. } else {
  2433. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2434. "A Phar stub cannot be set in a plain zip archive");
  2435. }
  2436. return;
  2437. }
  2438. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS() TSRMLS_CC, "r|l", &zstub, &len) == SUCCESS) {
  2439. if ((php_stream_from_zval_no_verify(stream, &zstub)) != NULL) {
  2440. if (len > 0) {
  2441. len = -len;
  2442. } else {
  2443. len = -1;
  2444. }
  2445. if (phar_obj->arc.archive->is_persistent && FAILURE == phar_copy_on_write(&(phar_obj->arc.archive) TSRMLS_CC)) {
  2446. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar_obj->arc.archive->fname);
  2447. return;
  2448. }
  2449. phar_flush(phar_obj->arc.archive, (char *) &zstub, len, 0, &error TSRMLS_CC);
  2450. if (error) {
  2451. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  2452. efree(error);
  2453. }
  2454. RETURN_TRUE;
  2455. } else {
  2456. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2457. "Cannot change stub, unable to read from input stream");
  2458. }
  2459. } else if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s", &stub, &stub_len) == SUCCESS) {
  2460. if (phar_obj->arc.archive->is_persistent && FAILURE == phar_copy_on_write(&(phar_obj->arc.archive) TSRMLS_CC)) {
  2461. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar_obj->arc.archive->fname);
  2462. return;
  2463. }
  2464. phar_flush(phar_obj->arc.archive, stub, stub_len, 0, &error TSRMLS_CC);
  2465. if (error) {
  2466. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  2467. efree(error);
  2468. }
  2469. RETURN_TRUE;
  2470. }
  2471. RETURN_FALSE;
  2472. }
  2473. /* }}} */
  2474. /* {{{ proto bool Phar::setDefaultStub([string index[, string webindex]])
  2475. * In a pure phar archive, sets a stub that can be used to run the archive
  2476. * regardless of whether the phar extension is available. The first parameter
  2477. * is the CLI startup filename, which defaults to "index.php". The second
  2478. * parameter is the web startup filename and also defaults to "index.php"
  2479. * (falling back to CLI behaviour).
  2480. * Both parameters are optional.
  2481. * In a phar.zip or phar.tar archive, the default stub is used only to
  2482. * identify the archive to the extension as a Phar object. This allows the
  2483. * extension to treat phar.zip and phar.tar types as honorary phars. Since
  2484. * files cannot be loaded via this kind of stub, no parameters are accepted
  2485. * when the Phar object is zip- or tar-based.
  2486. */
  2487. PHP_METHOD(Phar, setDefaultStub)
  2488. {
  2489. char *index = NULL, *webindex = NULL, *error = NULL, *stub = NULL;
  2490. int index_len = 0, webindex_len = 0, created_stub = 0;
  2491. size_t stub_len = 0;
  2492. PHAR_ARCHIVE_OBJECT();
  2493. if (phar_obj->arc.archive->is_data) {
  2494. if (phar_obj->arc.archive->is_tar) {
  2495. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2496. "A Phar stub cannot be set in a plain tar archive");
  2497. } else {
  2498. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2499. "A Phar stub cannot be set in a plain zip archive");
  2500. }
  2501. return;
  2502. }
  2503. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|s!s", &index, &index_len, &webindex, &webindex_len) == FAILURE) {
  2504. RETURN_FALSE;
  2505. }
  2506. if (ZEND_NUM_ARGS() > 0 && (phar_obj->arc.archive->is_tar || phar_obj->arc.archive->is_zip)) {
  2507. php_error_docref(NULL TSRMLS_CC, E_WARNING, "method accepts no arguments for a tar- or zip-based phar stub, %d given", ZEND_NUM_ARGS());
  2508. RETURN_FALSE;
  2509. }
  2510. if (PHAR_G(readonly)) {
  2511. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2512. "Cannot change stub: phar.readonly=1");
  2513. RETURN_FALSE;
  2514. }
  2515. if (!phar_obj->arc.archive->is_tar && !phar_obj->arc.archive->is_zip) {
  2516. stub = phar_create_default_stub(index, webindex, &stub_len, &error TSRMLS_CC);
  2517. if (error) {
  2518. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "%s", error);
  2519. efree(error);
  2520. if (stub) {
  2521. efree(stub);
  2522. }
  2523. RETURN_FALSE;
  2524. }
  2525. created_stub = 1;
  2526. }
  2527. if (phar_obj->arc.archive->is_persistent && FAILURE == phar_copy_on_write(&(phar_obj->arc.archive) TSRMLS_CC)) {
  2528. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar_obj->arc.archive->fname);
  2529. return;
  2530. }
  2531. phar_flush(phar_obj->arc.archive, stub, stub_len, 1, &error TSRMLS_CC);
  2532. if (created_stub) {
  2533. efree(stub);
  2534. }
  2535. if (error) {
  2536. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  2537. efree(error);
  2538. RETURN_FALSE;
  2539. }
  2540. RETURN_TRUE;
  2541. }
  2542. /* }}} */
  2543. /* {{{ proto array Phar::setSignatureAlgorithm(int sigtype[, string privatekey])
  2544. * Sets the signature algorithm for a phar and applies it. The signature
  2545. * algorithm must be one of Phar::MD5, Phar::SHA1, Phar::SHA256,
  2546. * Phar::SHA512, or Phar::OPENSSL. Note that zip- based phar archives
  2547. * cannot support signatures.
  2548. */
  2549. PHP_METHOD(Phar, setSignatureAlgorithm)
  2550. {
  2551. long algo;
  2552. char *error, *key = NULL;
  2553. int key_len = 0;
  2554. PHAR_ARCHIVE_OBJECT();
  2555. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  2556. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2557. "Cannot set signature algorithm, phar is read-only");
  2558. return;
  2559. }
  2560. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS() TSRMLS_CC, "l|s", &algo, &key, &key_len) != SUCCESS) {
  2561. return;
  2562. }
  2563. switch (algo) {
  2564. case PHAR_SIG_SHA256:
  2565. case PHAR_SIG_SHA512:
  2566. #ifndef PHAR_HASH_OK
  2567. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2568. "SHA-256 and SHA-512 signatures are only supported if the hash extension is enabled and built non-shared");
  2569. return;
  2570. #endif
  2571. case PHAR_SIG_MD5:
  2572. case PHAR_SIG_SHA1:
  2573. case PHAR_SIG_OPENSSL:
  2574. if (phar_obj->arc.archive->is_persistent && FAILURE == phar_copy_on_write(&(phar_obj->arc.archive) TSRMLS_CC)) {
  2575. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar_obj->arc.archive->fname);
  2576. return;
  2577. }
  2578. phar_obj->arc.archive->sig_flags = algo;
  2579. phar_obj->arc.archive->is_modified = 1;
  2580. PHAR_G(openssl_privatekey) = key;
  2581. PHAR_G(openssl_privatekey_len) = key_len;
  2582. phar_flush(phar_obj->arc.archive, 0, 0, 0, &error TSRMLS_CC);
  2583. if (error) {
  2584. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  2585. efree(error);
  2586. }
  2587. break;
  2588. default:
  2589. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2590. "Unknown signature algorithm specified");
  2591. }
  2592. }
  2593. /* }}} */
  2594. /* {{{ proto array|false Phar::getSignature()
  2595. * Returns a hash signature, or FALSE if the archive is unsigned.
  2596. */
  2597. PHP_METHOD(Phar, getSignature)
  2598. {
  2599. PHAR_ARCHIVE_OBJECT();
  2600. if (zend_parse_parameters_none() == FAILURE) {
  2601. return;
  2602. }
  2603. if (phar_obj->arc.archive->signature) {
  2604. char *unknown;
  2605. int unknown_len;
  2606. array_init(return_value);
  2607. add_assoc_stringl(return_value, "hash", phar_obj->arc.archive->signature, phar_obj->arc.archive->sig_len, 1);
  2608. switch(phar_obj->arc.archive->sig_flags) {
  2609. case PHAR_SIG_MD5:
  2610. add_assoc_stringl(return_value, "hash_type", "MD5", 3, 1);
  2611. break;
  2612. case PHAR_SIG_SHA1:
  2613. add_assoc_stringl(return_value, "hash_type", "SHA-1", 5, 1);
  2614. break;
  2615. case PHAR_SIG_SHA256:
  2616. add_assoc_stringl(return_value, "hash_type", "SHA-256", 7, 1);
  2617. break;
  2618. case PHAR_SIG_SHA512:
  2619. add_assoc_stringl(return_value, "hash_type", "SHA-512", 7, 1);
  2620. break;
  2621. case PHAR_SIG_OPENSSL:
  2622. add_assoc_stringl(return_value, "hash_type", "OpenSSL", 7, 1);
  2623. break;
  2624. default:
  2625. unknown_len = spprintf(&unknown, 0, "Unknown (%u)", phar_obj->arc.archive->sig_flags);
  2626. add_assoc_stringl(return_value, "hash_type", unknown, unknown_len, 0);
  2627. break;
  2628. }
  2629. } else {
  2630. RETURN_FALSE;
  2631. }
  2632. }
  2633. /* }}} */
  2634. /* {{{ proto bool Phar::getModified()
  2635. * Return whether phar was modified
  2636. */
  2637. PHP_METHOD(Phar, getModified)
  2638. {
  2639. PHAR_ARCHIVE_OBJECT();
  2640. if (zend_parse_parameters_none() == FAILURE) {
  2641. return;
  2642. }
  2643. RETURN_BOOL(phar_obj->arc.archive->is_modified);
  2644. }
  2645. /* }}} */
  2646. static int phar_set_compression(void *pDest, void *argument TSRMLS_DC) /* {{{ */
  2647. {
  2648. phar_entry_info *entry = (phar_entry_info *)pDest;
  2649. php_uint32 compress = *(php_uint32 *)argument;
  2650. if (entry->is_deleted) {
  2651. return ZEND_HASH_APPLY_KEEP;
  2652. }
  2653. entry->old_flags = entry->flags;
  2654. entry->flags &= ~PHAR_ENT_COMPRESSION_MASK;
  2655. entry->flags |= compress;
  2656. entry->is_modified = 1;
  2657. return ZEND_HASH_APPLY_KEEP;
  2658. }
  2659. /* }}} */
  2660. static int phar_test_compression(void *pDest, void *argument TSRMLS_DC) /* {{{ */
  2661. {
  2662. phar_entry_info *entry = (phar_entry_info *)pDest;
  2663. if (entry->is_deleted) {
  2664. return ZEND_HASH_APPLY_KEEP;
  2665. }
  2666. if (!PHAR_G(has_bz2)) {
  2667. if (entry->flags & PHAR_ENT_COMPRESSED_BZ2) {
  2668. *(int *) argument = 0;
  2669. }
  2670. }
  2671. if (!PHAR_G(has_zlib)) {
  2672. if (entry->flags & PHAR_ENT_COMPRESSED_GZ) {
  2673. *(int *) argument = 0;
  2674. }
  2675. }
  2676. return ZEND_HASH_APPLY_KEEP;
  2677. }
  2678. /* }}} */
  2679. static void pharobj_set_compression(HashTable *manifest, php_uint32 compress TSRMLS_DC) /* {{{ */
  2680. {
  2681. zend_hash_apply_with_argument(manifest, phar_set_compression, &compress TSRMLS_CC);
  2682. }
  2683. /* }}} */
  2684. static int pharobj_cancompress(HashTable *manifest TSRMLS_DC) /* {{{ */
  2685. {
  2686. int test;
  2687. test = 1;
  2688. zend_hash_apply_with_argument(manifest, phar_test_compression, &test TSRMLS_CC);
  2689. return test;
  2690. }
  2691. /* }}} */
  2692. /* {{{ proto object Phar::compress(int method[, string extension])
  2693. * Compress a .tar, or .phar.tar with whole-file compression
  2694. * The parameter can be one of Phar::GZ or Phar::BZ2 to specify
  2695. * the kind of compression desired
  2696. */
  2697. PHP_METHOD(Phar, compress)
  2698. {
  2699. long method;
  2700. char *ext = NULL;
  2701. int ext_len = 0;
  2702. php_uint32 flags;
  2703. zval *ret;
  2704. PHAR_ARCHIVE_OBJECT();
  2705. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l|s", &method, &ext, &ext_len) == FAILURE) {
  2706. return;
  2707. }
  2708. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  2709. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2710. "Cannot compress phar archive, phar is read-only");
  2711. return;
  2712. }
  2713. if (phar_obj->arc.archive->is_zip) {
  2714. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2715. "Cannot compress zip-based archives with whole-archive compression");
  2716. return;
  2717. }
  2718. switch (method) {
  2719. case 0:
  2720. flags = PHAR_FILE_COMPRESSED_NONE;
  2721. break;
  2722. case PHAR_ENT_COMPRESSED_GZ:
  2723. if (!PHAR_G(has_zlib)) {
  2724. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2725. "Cannot compress entire archive with gzip, enable ext/zlib in php.ini");
  2726. return;
  2727. }
  2728. flags = PHAR_FILE_COMPRESSED_GZ;
  2729. break;
  2730. case PHAR_ENT_COMPRESSED_BZ2:
  2731. if (!PHAR_G(has_bz2)) {
  2732. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2733. "Cannot compress entire archive with bz2, enable ext/bz2 in php.ini");
  2734. return;
  2735. }
  2736. flags = PHAR_FILE_COMPRESSED_BZ2;
  2737. break;
  2738. default:
  2739. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2740. "Unknown compression specified, please pass one of Phar::GZ or Phar::BZ2");
  2741. return;
  2742. }
  2743. if (phar_obj->arc.archive->is_tar) {
  2744. ret = phar_convert_to_other(phar_obj->arc.archive, PHAR_FORMAT_TAR, ext, flags TSRMLS_CC);
  2745. } else {
  2746. ret = phar_convert_to_other(phar_obj->arc.archive, PHAR_FORMAT_PHAR, ext, flags TSRMLS_CC);
  2747. }
  2748. if (ret) {
  2749. RETURN_ZVAL(ret, 1, 1);
  2750. } else {
  2751. RETURN_NULL();
  2752. }
  2753. }
  2754. /* }}} */
  2755. /* {{{ proto object Phar::decompress([string extension])
  2756. * Decompress a .tar, or .phar.tar with whole-file compression
  2757. */
  2758. PHP_METHOD(Phar, decompress)
  2759. {
  2760. char *ext = NULL;
  2761. int ext_len = 0;
  2762. zval *ret;
  2763. PHAR_ARCHIVE_OBJECT();
  2764. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|s", &ext, &ext_len) == FAILURE) {
  2765. return;
  2766. }
  2767. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  2768. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2769. "Cannot decompress phar archive, phar is read-only");
  2770. return;
  2771. }
  2772. if (phar_obj->arc.archive->is_zip) {
  2773. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2774. "Cannot decompress zip-based archives with whole-archive compression");
  2775. return;
  2776. }
  2777. if (phar_obj->arc.archive->is_tar) {
  2778. ret = phar_convert_to_other(phar_obj->arc.archive, PHAR_FORMAT_TAR, ext, PHAR_FILE_COMPRESSED_NONE TSRMLS_CC);
  2779. } else {
  2780. ret = phar_convert_to_other(phar_obj->arc.archive, PHAR_FORMAT_PHAR, ext, PHAR_FILE_COMPRESSED_NONE TSRMLS_CC);
  2781. }
  2782. if (ret) {
  2783. RETURN_ZVAL(ret, 1, 1);
  2784. } else {
  2785. RETURN_NULL();
  2786. }
  2787. }
  2788. /* }}} */
  2789. /* {{{ proto object Phar::compressFiles(int method)
  2790. * Compress all files within a phar or zip archive using the specified compression
  2791. * The parameter can be one of Phar::GZ or Phar::BZ2 to specify
  2792. * the kind of compression desired
  2793. */
  2794. PHP_METHOD(Phar, compressFiles)
  2795. {
  2796. char *error;
  2797. php_uint32 flags;
  2798. long method;
  2799. PHAR_ARCHIVE_OBJECT();
  2800. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &method) == FAILURE) {
  2801. return;
  2802. }
  2803. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  2804. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2805. "Phar is readonly, cannot change compression");
  2806. return;
  2807. }
  2808. switch (method) {
  2809. case PHAR_ENT_COMPRESSED_GZ:
  2810. if (!PHAR_G(has_zlib)) {
  2811. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2812. "Cannot compress files within archive with gzip, enable ext/zlib in php.ini");
  2813. return;
  2814. }
  2815. flags = PHAR_ENT_COMPRESSED_GZ;
  2816. break;
  2817. case PHAR_ENT_COMPRESSED_BZ2:
  2818. if (!PHAR_G(has_bz2)) {
  2819. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2820. "Cannot compress files within archive with bz2, enable ext/bz2 in php.ini");
  2821. return;
  2822. }
  2823. flags = PHAR_ENT_COMPRESSED_BZ2;
  2824. break;
  2825. default:
  2826. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2827. "Unknown compression specified, please pass one of Phar::GZ or Phar::BZ2");
  2828. return;
  2829. }
  2830. if (phar_obj->arc.archive->is_tar) {
  2831. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2832. "Cannot compress with Gzip compression, tar archives cannot compress individual files, use compress() to compress the whole archive");
  2833. return;
  2834. }
  2835. if (!pharobj_cancompress(&phar_obj->arc.archive->manifest TSRMLS_CC)) {
  2836. if (flags == PHAR_FILE_COMPRESSED_GZ) {
  2837. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2838. "Cannot compress all files as Gzip, some are compressed as bzip2 and cannot be decompressed");
  2839. } else {
  2840. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2841. "Cannot compress all files as Bzip2, some are compressed as gzip and cannot be decompressed");
  2842. }
  2843. return;
  2844. }
  2845. if (phar_obj->arc.archive->is_persistent && FAILURE == phar_copy_on_write(&(phar_obj->arc.archive) TSRMLS_CC)) {
  2846. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar_obj->arc.archive->fname);
  2847. return;
  2848. }
  2849. pharobj_set_compression(&phar_obj->arc.archive->manifest, flags TSRMLS_CC);
  2850. phar_obj->arc.archive->is_modified = 1;
  2851. phar_flush(phar_obj->arc.archive, 0, 0, 0, &error TSRMLS_CC);
  2852. if (error) {
  2853. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "%s", error);
  2854. efree(error);
  2855. }
  2856. }
  2857. /* }}} */
  2858. /* {{{ proto bool Phar::decompressFiles()
  2859. * decompress every file
  2860. */
  2861. PHP_METHOD(Phar, decompressFiles)
  2862. {
  2863. char *error;
  2864. PHAR_ARCHIVE_OBJECT();
  2865. if (zend_parse_parameters_none() == FAILURE) {
  2866. return;
  2867. }
  2868. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  2869. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2870. "Phar is readonly, cannot change compression");
  2871. return;
  2872. }
  2873. if (!pharobj_cancompress(&phar_obj->arc.archive->manifest TSRMLS_CC)) {
  2874. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  2875. "Cannot decompress all files, some are compressed as bzip2 or gzip and cannot be decompressed");
  2876. return;
  2877. }
  2878. if (phar_obj->arc.archive->is_tar) {
  2879. RETURN_TRUE;
  2880. } else {
  2881. if (phar_obj->arc.archive->is_persistent && FAILURE == phar_copy_on_write(&(phar_obj->arc.archive) TSRMLS_CC)) {
  2882. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar_obj->arc.archive->fname);
  2883. return;
  2884. }
  2885. pharobj_set_compression(&phar_obj->arc.archive->manifest, PHAR_ENT_COMPRESSED_NONE TSRMLS_CC);
  2886. }
  2887. phar_obj->arc.archive->is_modified = 1;
  2888. phar_flush(phar_obj->arc.archive, 0, 0, 0, &error TSRMLS_CC);
  2889. if (error) {
  2890. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "%s", error);
  2891. efree(error);
  2892. }
  2893. RETURN_TRUE;
  2894. }
  2895. /* }}} */
  2896. /* {{{ proto bool Phar::copy(string oldfile, string newfile)
  2897. * copy a file internal to the phar archive to another new file within the phar
  2898. */
  2899. PHP_METHOD(Phar, copy)
  2900. {
  2901. char *oldfile, *newfile, *error;
  2902. const char *pcr_error;
  2903. int oldfile_len, newfile_len;
  2904. phar_entry_info *oldentry, newentry = {0}, *temp;
  2905. PHAR_ARCHIVE_OBJECT();
  2906. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "pp", &oldfile, &oldfile_len, &newfile, &newfile_len) == FAILURE) {
  2907. return;
  2908. }
  2909. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  2910. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2911. "Cannot copy \"%s\" to \"%s\", phar is read-only", oldfile, newfile);
  2912. RETURN_FALSE;
  2913. }
  2914. if (oldfile_len >= sizeof(".phar")-1 && !memcmp(oldfile, ".phar", sizeof(".phar")-1)) {
  2915. /* can't copy a meta file */
  2916. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2917. "file \"%s\" cannot be copied to file \"%s\", cannot copy Phar meta-file in %s", oldfile, newfile, phar_obj->arc.archive->fname);
  2918. RETURN_FALSE;
  2919. }
  2920. if (newfile_len >= sizeof(".phar")-1 && !memcmp(newfile, ".phar", sizeof(".phar")-1)) {
  2921. /* can't copy a meta file */
  2922. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2923. "file \"%s\" cannot be copied to file \"%s\", cannot copy to Phar meta-file in %s", oldfile, newfile, phar_obj->arc.archive->fname);
  2924. RETURN_FALSE;
  2925. }
  2926. if (!zend_hash_exists(&phar_obj->arc.archive->manifest, oldfile, (uint) oldfile_len) || SUCCESS != zend_hash_find(&phar_obj->arc.archive->manifest, oldfile, (uint) oldfile_len, (void**)&oldentry) || oldentry->is_deleted) {
  2927. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2928. "file \"%s\" cannot be copied to file \"%s\", file does not exist in %s", oldfile, newfile, phar_obj->arc.archive->fname);
  2929. RETURN_FALSE;
  2930. }
  2931. if (zend_hash_exists(&phar_obj->arc.archive->manifest, newfile, (uint) newfile_len)) {
  2932. if (SUCCESS == zend_hash_find(&phar_obj->arc.archive->manifest, newfile, (uint) newfile_len, (void**)&temp) || !temp->is_deleted) {
  2933. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2934. "file \"%s\" cannot be copied to file \"%s\", file must not already exist in phar %s", oldfile, newfile, phar_obj->arc.archive->fname);
  2935. RETURN_FALSE;
  2936. }
  2937. }
  2938. if (phar_path_check(&newfile, &newfile_len, &pcr_error) > pcr_is_ok) {
  2939. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC,
  2940. "file \"%s\" contains invalid characters %s, cannot be copied from \"%s\" in phar %s", newfile, pcr_error, oldfile, phar_obj->arc.archive->fname);
  2941. RETURN_FALSE;
  2942. }
  2943. if (phar_obj->arc.archive->is_persistent) {
  2944. if (FAILURE == phar_copy_on_write(&(phar_obj->arc.archive) TSRMLS_CC)) {
  2945. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar_obj->arc.archive->fname);
  2946. return;
  2947. }
  2948. /* re-populate with copied-on-write entry */
  2949. zend_hash_find(&phar_obj->arc.archive->manifest, oldfile, (uint) oldfile_len, (void**)&oldentry);
  2950. }
  2951. memcpy((void *) &newentry, oldentry, sizeof(phar_entry_info));
  2952. if (newentry.metadata) {
  2953. zval *t;
  2954. t = newentry.metadata;
  2955. ALLOC_ZVAL(newentry.metadata);
  2956. *newentry.metadata = *t;
  2957. zval_copy_ctor(newentry.metadata);
  2958. Z_SET_REFCOUNT_P(newentry.metadata, 1);
  2959. newentry.metadata_str.c = NULL;
  2960. newentry.metadata_str.len = 0;
  2961. }
  2962. newentry.filename = estrndup(newfile, newfile_len);
  2963. newentry.filename_len = newfile_len;
  2964. newentry.fp_refcount = 0;
  2965. if (oldentry->fp_type != PHAR_FP) {
  2966. if (FAILURE == phar_copy_entry_fp(oldentry, &newentry, &error TSRMLS_CC)) {
  2967. efree(newentry.filename);
  2968. php_stream_close(newentry.fp);
  2969. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  2970. efree(error);
  2971. return;
  2972. }
  2973. }
  2974. zend_hash_add(&oldentry->phar->manifest, newfile, newfile_len, (void*)&newentry, sizeof(phar_entry_info), NULL);
  2975. phar_obj->arc.archive->is_modified = 1;
  2976. phar_flush(phar_obj->arc.archive, 0, 0, 0, &error TSRMLS_CC);
  2977. if (error) {
  2978. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  2979. efree(error);
  2980. }
  2981. RETURN_TRUE;
  2982. }
  2983. /* }}} */
  2984. /* {{{ proto int Phar::offsetExists(string entry)
  2985. * determines whether a file exists in the phar
  2986. */
  2987. PHP_METHOD(Phar, offsetExists)
  2988. {
  2989. char *fname;
  2990. int fname_len;
  2991. phar_entry_info *entry;
  2992. PHAR_ARCHIVE_OBJECT();
  2993. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p", &fname, &fname_len) == FAILURE) {
  2994. return;
  2995. }
  2996. if (zend_hash_exists(&phar_obj->arc.archive->manifest, fname, (uint) fname_len)) {
  2997. if (SUCCESS == zend_hash_find(&phar_obj->arc.archive->manifest, fname, (uint) fname_len, (void**)&entry)) {
  2998. if (entry->is_deleted) {
  2999. /* entry is deleted, but has not been flushed to disk yet */
  3000. RETURN_FALSE;
  3001. }
  3002. }
  3003. if (fname_len >= sizeof(".phar")-1 && !memcmp(fname, ".phar", sizeof(".phar")-1)) {
  3004. /* none of these are real files, so they don't exist */
  3005. RETURN_FALSE;
  3006. }
  3007. RETURN_TRUE;
  3008. } else {
  3009. if (zend_hash_exists(&phar_obj->arc.archive->virtual_dirs, fname, (uint) fname_len)) {
  3010. RETURN_TRUE;
  3011. }
  3012. RETURN_FALSE;
  3013. }
  3014. }
  3015. /* }}} */
  3016. /* {{{ proto int Phar::offsetGet(string entry)
  3017. * get a PharFileInfo object for a specific file
  3018. */
  3019. PHP_METHOD(Phar, offsetGet)
  3020. {
  3021. char *fname, *error;
  3022. int fname_len;
  3023. zval *zfname;
  3024. phar_entry_info *entry;
  3025. PHAR_ARCHIVE_OBJECT();
  3026. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p", &fname, &fname_len) == FAILURE) {
  3027. return;
  3028. }
  3029. /* security is 0 here so that we can get a better error message than "entry doesn't exist" */
  3030. if (!(entry = phar_get_entry_info_dir(phar_obj->arc.archive, fname, fname_len, 1, &error, 0 TSRMLS_CC))) {
  3031. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Entry %s does not exist%s%s", fname, error?", ":"", error?error:"");
  3032. } else {
  3033. if (fname_len == sizeof(".phar/stub.php")-1 && !memcmp(fname, ".phar/stub.php", sizeof(".phar/stub.php")-1)) {
  3034. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Cannot get stub \".phar/stub.php\" directly in phar \"%s\", use getStub", phar_obj->arc.archive->fname);
  3035. return;
  3036. }
  3037. if (fname_len == sizeof(".phar/alias.txt")-1 && !memcmp(fname, ".phar/alias.txt", sizeof(".phar/alias.txt")-1)) {
  3038. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Cannot get alias \".phar/alias.txt\" directly in phar \"%s\", use getAlias", phar_obj->arc.archive->fname);
  3039. return;
  3040. }
  3041. if (fname_len >= sizeof(".phar")-1 && !memcmp(fname, ".phar", sizeof(".phar")-1)) {
  3042. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Cannot directly get any files or directories in magic \".phar\" directory", phar_obj->arc.archive->fname);
  3043. return;
  3044. }
  3045. if (entry->is_temp_dir) {
  3046. efree(entry->filename);
  3047. efree(entry);
  3048. }
  3049. fname_len = spprintf(&fname, 0, "phar://%s/%s", phar_obj->arc.archive->fname, fname);
  3050. MAKE_STD_ZVAL(zfname);
  3051. ZVAL_STRINGL(zfname, fname, fname_len, 0);
  3052. spl_instantiate_arg_ex1(phar_obj->spl.info_class, &return_value, 0, zfname TSRMLS_CC);
  3053. zval_ptr_dtor(&zfname);
  3054. }
  3055. }
  3056. /* }}} */
  3057. /* {{{ add a file within the phar archive from a string or resource
  3058. */
  3059. static void phar_add_file(phar_archive_data **pphar, char *filename, int filename_len, char *cont_str, int cont_len, zval *zresource TSRMLS_DC)
  3060. {
  3061. char *error;
  3062. size_t contents_len;
  3063. phar_entry_data *data;
  3064. php_stream *contents_file = NULL;
  3065. php_stream_statbuf ssb;
  3066. if (filename_len >= sizeof(".phar")-1 && !memcmp(filename, ".phar", sizeof(".phar")-1) && (filename[5] == '/' || filename[5] == '\\' || filename[5] == '\0')) {
  3067. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Cannot create any files in magic \".phar\" directory", (*pphar)->fname);
  3068. return;
  3069. }
  3070. if (!(data = phar_get_or_create_entry_data((*pphar)->fname, (*pphar)->fname_len, filename, filename_len, "w+b", 0, &error, 1 TSRMLS_CC))) {
  3071. if (error) {
  3072. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Entry %s does not exist and cannot be created: %s", filename, error);
  3073. efree(error);
  3074. } else {
  3075. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Entry %s does not exist and cannot be created", filename);
  3076. }
  3077. return;
  3078. } else {
  3079. if (error) {
  3080. efree(error);
  3081. }
  3082. if (!data->internal_file->is_dir) {
  3083. if (cont_str) {
  3084. contents_len = php_stream_write(data->fp, cont_str, cont_len);
  3085. if (contents_len != cont_len) {
  3086. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Entry %s could not be written to", filename);
  3087. return;
  3088. }
  3089. } else {
  3090. if (!(php_stream_from_zval_no_verify(contents_file, &zresource))) {
  3091. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Entry %s could not be written to", filename);
  3092. return;
  3093. }
  3094. php_stream_copy_to_stream_ex(contents_file, data->fp, PHP_STREAM_COPY_ALL, &contents_len);
  3095. }
  3096. data->internal_file->compressed_filesize = data->internal_file->uncompressed_filesize = contents_len;
  3097. }
  3098. if (contents_file != NULL && php_stream_stat(contents_file, &ssb) != -1) {
  3099. data->internal_file->flags = ssb.sb.st_mode & PHAR_ENT_PERM_MASK ;
  3100. } else {
  3101. #ifndef _WIN32
  3102. mode_t mask;
  3103. mask = umask(0);
  3104. umask(mask);
  3105. data->internal_file->flags &= ~mask;
  3106. #endif
  3107. }
  3108. /* check for copy-on-write */
  3109. if (pphar[0] != data->phar) {
  3110. *pphar = data->phar;
  3111. }
  3112. phar_entry_delref(data TSRMLS_CC);
  3113. phar_flush(*pphar, 0, 0, 0, &error TSRMLS_CC);
  3114. if (error) {
  3115. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  3116. efree(error);
  3117. }
  3118. }
  3119. }
  3120. /* }}} */
  3121. /* {{{ create a directory within the phar archive
  3122. */
  3123. static void phar_mkdir(phar_archive_data **pphar, char *dirname, int dirname_len TSRMLS_DC)
  3124. {
  3125. char *error;
  3126. phar_entry_data *data;
  3127. if (!(data = phar_get_or_create_entry_data((*pphar)->fname, (*pphar)->fname_len, dirname, dirname_len, "w+b", 2, &error, 1 TSRMLS_CC))) {
  3128. if (error) {
  3129. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Directory %s does not exist and cannot be created: %s", dirname, error);
  3130. efree(error);
  3131. } else {
  3132. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Directory %s does not exist and cannot be created", dirname);
  3133. }
  3134. return;
  3135. } else {
  3136. if (error) {
  3137. efree(error);
  3138. }
  3139. /* check for copy on write */
  3140. if (data->phar != *pphar) {
  3141. *pphar = data->phar;
  3142. }
  3143. phar_entry_delref(data TSRMLS_CC);
  3144. phar_flush(*pphar, 0, 0, 0, &error TSRMLS_CC);
  3145. if (error) {
  3146. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  3147. efree(error);
  3148. }
  3149. }
  3150. }
  3151. /* }}} */
  3152. /* {{{ proto int Phar::offsetSet(string entry, string value)
  3153. * set the contents of an internal file to those of an external file
  3154. */
  3155. PHP_METHOD(Phar, offsetSet)
  3156. {
  3157. char *fname, *cont_str = NULL;
  3158. int fname_len, cont_len;
  3159. zval *zresource;
  3160. PHAR_ARCHIVE_OBJECT();
  3161. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  3162. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Write operations disabled by the php.ini setting phar.readonly");
  3163. return;
  3164. }
  3165. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS() TSRMLS_CC, "pr", &fname, &fname_len, &zresource) == FAILURE
  3166. && zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ps", &fname, &fname_len, &cont_str, &cont_len) == FAILURE) {
  3167. return;
  3168. }
  3169. if (fname_len == sizeof(".phar/stub.php")-1 && !memcmp(fname, ".phar/stub.php", sizeof(".phar/stub.php")-1)) {
  3170. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Cannot set stub \".phar/stub.php\" directly in phar \"%s\", use setStub", phar_obj->arc.archive->fname);
  3171. return;
  3172. }
  3173. if (fname_len == sizeof(".phar/alias.txt")-1 && !memcmp(fname, ".phar/alias.txt", sizeof(".phar/alias.txt")-1)) {
  3174. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Cannot set alias \".phar/alias.txt\" directly in phar \"%s\", use setAlias", phar_obj->arc.archive->fname);
  3175. return;
  3176. }
  3177. if (fname_len >= sizeof(".phar")-1 && !memcmp(fname, ".phar", sizeof(".phar")-1)) {
  3178. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Cannot set any files or directories in magic \".phar\" directory", phar_obj->arc.archive->fname);
  3179. return;
  3180. }
  3181. phar_add_file(&(phar_obj->arc.archive), fname, fname_len, cont_str, cont_len, zresource TSRMLS_CC);
  3182. }
  3183. /* }}} */
  3184. /* {{{ proto int Phar::offsetUnset(string entry)
  3185. * remove a file from a phar
  3186. */
  3187. PHP_METHOD(Phar, offsetUnset)
  3188. {
  3189. char *fname, *error;
  3190. int fname_len;
  3191. phar_entry_info *entry;
  3192. PHAR_ARCHIVE_OBJECT();
  3193. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  3194. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Write operations disabled by the php.ini setting phar.readonly");
  3195. return;
  3196. }
  3197. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p", &fname, &fname_len) == FAILURE) {
  3198. return;
  3199. }
  3200. if (zend_hash_exists(&phar_obj->arc.archive->manifest, fname, (uint) fname_len)) {
  3201. if (SUCCESS == zend_hash_find(&phar_obj->arc.archive->manifest, fname, (uint) fname_len, (void**)&entry)) {
  3202. if (entry->is_deleted) {
  3203. /* entry is deleted, but has not been flushed to disk yet */
  3204. return;
  3205. }
  3206. if (phar_obj->arc.archive->is_persistent) {
  3207. if (FAILURE == phar_copy_on_write(&(phar_obj->arc.archive) TSRMLS_CC)) {
  3208. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar_obj->arc.archive->fname);
  3209. return;
  3210. }
  3211. /* re-populate entry after copy on write */
  3212. zend_hash_find(&phar_obj->arc.archive->manifest, fname, (uint) fname_len, (void **)&entry);
  3213. }
  3214. entry->is_modified = 0;
  3215. entry->is_deleted = 1;
  3216. /* we need to "flush" the stream to save the newly deleted file on disk */
  3217. phar_flush(phar_obj->arc.archive, 0, 0, 0, &error TSRMLS_CC);
  3218. if (error) {
  3219. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  3220. efree(error);
  3221. }
  3222. RETURN_TRUE;
  3223. }
  3224. } else {
  3225. RETURN_FALSE;
  3226. }
  3227. }
  3228. /* }}} */
  3229. /* {{{ proto string Phar::addEmptyDir(string dirname)
  3230. * Adds an empty directory to the phar archive
  3231. */
  3232. PHP_METHOD(Phar, addEmptyDir)
  3233. {
  3234. char *dirname;
  3235. int dirname_len;
  3236. PHAR_ARCHIVE_OBJECT();
  3237. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p", &dirname, &dirname_len) == FAILURE) {
  3238. return;
  3239. }
  3240. if (dirname_len >= sizeof(".phar")-1 && !memcmp(dirname, ".phar", sizeof(".phar")-1)) {
  3241. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Cannot create a directory in magic \".phar\" directory");
  3242. return;
  3243. }
  3244. phar_mkdir(&phar_obj->arc.archive, dirname, dirname_len TSRMLS_CC);
  3245. }
  3246. /* }}} */
  3247. /* {{{ proto string Phar::addFile(string filename[, string localname])
  3248. * Adds a file to the archive using the filename, or the second parameter as the name within the archive
  3249. */
  3250. PHP_METHOD(Phar, addFile)
  3251. {
  3252. char *fname, *localname = NULL;
  3253. int fname_len, localname_len = 0;
  3254. php_stream *resource;
  3255. zval *zresource;
  3256. PHAR_ARCHIVE_OBJECT();
  3257. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p|s", &fname, &fname_len, &localname, &localname_len) == FAILURE) {
  3258. return;
  3259. }
  3260. #if PHP_API_VERSION < 20100412
  3261. if (PG(safe_mode) && (!php_checkuid(fname, NULL, CHECKUID_ALLOW_ONLY_FILE))) {
  3262. zend_throw_exception_ex(spl_ce_RuntimeException, 0 TSRMLS_CC, "phar error: unable to open file \"%s\" to add to phar archive, safe_mode restrictions prevent this", fname);
  3263. return;
  3264. }
  3265. #endif
  3266. if (!strstr(fname, "://") && php_check_open_basedir(fname TSRMLS_CC)) {
  3267. zend_throw_exception_ex(spl_ce_RuntimeException, 0 TSRMLS_CC, "phar error: unable to open file \"%s\" to add to phar archive, open_basedir restrictions prevent this", fname);
  3268. return;
  3269. }
  3270. if (!(resource = php_stream_open_wrapper(fname, "rb", 0, NULL))) {
  3271. zend_throw_exception_ex(spl_ce_RuntimeException, 0 TSRMLS_CC, "phar error: unable to open file \"%s\" to add to phar archive", fname);
  3272. return;
  3273. }
  3274. if (localname) {
  3275. fname = localname;
  3276. fname_len = localname_len;
  3277. }
  3278. MAKE_STD_ZVAL(zresource);
  3279. php_stream_to_zval(resource, zresource);
  3280. phar_add_file(&(phar_obj->arc.archive), fname, fname_len, NULL, 0, zresource TSRMLS_CC);
  3281. efree(zresource);
  3282. php_stream_close(resource);
  3283. }
  3284. /* }}} */
  3285. /* {{{ proto string Phar::addFromString(string localname, string contents)
  3286. * Adds a file to the archive using its contents as a string
  3287. */
  3288. PHP_METHOD(Phar, addFromString)
  3289. {
  3290. char *localname, *cont_str;
  3291. int localname_len, cont_len;
  3292. PHAR_ARCHIVE_OBJECT();
  3293. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ps", &localname, &localname_len, &cont_str, &cont_len) == FAILURE) {
  3294. return;
  3295. }
  3296. phar_add_file(&(phar_obj->arc.archive), localname, localname_len, cont_str, cont_len, NULL TSRMLS_CC);
  3297. }
  3298. /* }}} */
  3299. /* {{{ proto string Phar::getStub()
  3300. * Returns the stub at the head of a phar archive as a string.
  3301. */
  3302. PHP_METHOD(Phar, getStub)
  3303. {
  3304. size_t len;
  3305. char *buf;
  3306. php_stream *fp;
  3307. php_stream_filter *filter = NULL;
  3308. phar_entry_info *stub;
  3309. PHAR_ARCHIVE_OBJECT();
  3310. if (zend_parse_parameters_none() == FAILURE) {
  3311. return;
  3312. }
  3313. if (phar_obj->arc.archive->is_tar || phar_obj->arc.archive->is_zip) {
  3314. if (SUCCESS == zend_hash_find(&(phar_obj->arc.archive->manifest), ".phar/stub.php", sizeof(".phar/stub.php")-1, (void **)&stub)) {
  3315. if (phar_obj->arc.archive->fp && !phar_obj->arc.archive->is_brandnew && !(stub->flags & PHAR_ENT_COMPRESSION_MASK)) {
  3316. fp = phar_obj->arc.archive->fp;
  3317. } else {
  3318. if (!(fp = php_stream_open_wrapper(phar_obj->arc.archive->fname, "rb", 0, NULL))) {
  3319. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "phar error: unable to open phar \"%s\"", phar_obj->arc.archive->fname);
  3320. return;
  3321. }
  3322. if (stub->flags & PHAR_ENT_COMPRESSION_MASK) {
  3323. char *filter_name;
  3324. if ((filter_name = phar_decompress_filter(stub, 0)) != NULL) {
  3325. filter = php_stream_filter_create(filter_name, NULL, php_stream_is_persistent(fp) TSRMLS_CC);
  3326. } else {
  3327. filter = NULL;
  3328. }
  3329. if (!filter) {
  3330. zend_throw_exception_ex(spl_ce_UnexpectedValueException, 0 TSRMLS_CC, "phar error: unable to read stub of phar \"%s\" (cannot create %s filter)", phar_obj->arc.archive->fname, phar_decompress_filter(stub, 1));
  3331. return;
  3332. }
  3333. php_stream_filter_append(&fp->readfilters, filter);
  3334. }
  3335. }
  3336. if (!fp) {
  3337. zend_throw_exception_ex(spl_ce_RuntimeException, 0 TSRMLS_CC,
  3338. "Unable to read stub");
  3339. return;
  3340. }
  3341. php_stream_seek(fp, stub->offset_abs, SEEK_SET);
  3342. len = stub->uncompressed_filesize;
  3343. goto carry_on;
  3344. } else {
  3345. RETURN_STRINGL("", 0, 1);
  3346. }
  3347. }
  3348. len = phar_obj->arc.archive->halt_offset;
  3349. if (phar_obj->arc.archive->fp && !phar_obj->arc.archive->is_brandnew) {
  3350. fp = phar_obj->arc.archive->fp;
  3351. } else {
  3352. fp = php_stream_open_wrapper(phar_obj->arc.archive->fname, "rb", 0, NULL);
  3353. }
  3354. if (!fp) {
  3355. zend_throw_exception_ex(spl_ce_RuntimeException, 0 TSRMLS_CC,
  3356. "Unable to read stub");
  3357. return;
  3358. }
  3359. php_stream_rewind(fp);
  3360. carry_on:
  3361. buf = safe_emalloc(len, 1, 1);
  3362. if (len != php_stream_read(fp, buf, len)) {
  3363. if (fp != phar_obj->arc.archive->fp) {
  3364. php_stream_close(fp);
  3365. }
  3366. zend_throw_exception_ex(spl_ce_RuntimeException, 0 TSRMLS_CC,
  3367. "Unable to read stub");
  3368. efree(buf);
  3369. return;
  3370. }
  3371. if (filter) {
  3372. php_stream_filter_flush(filter, 1);
  3373. php_stream_filter_remove(filter, 1 TSRMLS_CC);
  3374. }
  3375. if (fp != phar_obj->arc.archive->fp) {
  3376. php_stream_close(fp);
  3377. }
  3378. buf[len] = '\0';
  3379. RETURN_STRINGL(buf, len, 0);
  3380. }
  3381. /* }}}*/
  3382. /* {{{ proto int Phar::hasMetaData()
  3383. * Returns TRUE if the phar has global metadata, FALSE otherwise.
  3384. */
  3385. PHP_METHOD(Phar, hasMetadata)
  3386. {
  3387. PHAR_ARCHIVE_OBJECT();
  3388. RETURN_BOOL(phar_obj->arc.archive->metadata != NULL);
  3389. }
  3390. /* }}} */
  3391. /* {{{ proto int Phar::getMetaData()
  3392. * Returns the global metadata of the phar
  3393. */
  3394. PHP_METHOD(Phar, getMetadata)
  3395. {
  3396. PHAR_ARCHIVE_OBJECT();
  3397. if (zend_parse_parameters_none() == FAILURE) {
  3398. return;
  3399. }
  3400. if (phar_obj->arc.archive->metadata) {
  3401. if (phar_obj->arc.archive->is_persistent) {
  3402. zval *ret;
  3403. char *buf = estrndup((char *) phar_obj->arc.archive->metadata, phar_obj->arc.archive->metadata_len);
  3404. /* assume success, we would have failed before */
  3405. phar_parse_metadata(&buf, &ret, phar_obj->arc.archive->metadata_len TSRMLS_CC);
  3406. efree(buf);
  3407. RETURN_ZVAL(ret, 0, 1);
  3408. }
  3409. RETURN_ZVAL(phar_obj->arc.archive->metadata, 1, 0);
  3410. }
  3411. }
  3412. /* }}} */
  3413. /* {{{ proto int Phar::setMetaData(mixed $metadata)
  3414. * Sets the global metadata of the phar
  3415. */
  3416. PHP_METHOD(Phar, setMetadata)
  3417. {
  3418. char *error;
  3419. zval *metadata;
  3420. PHAR_ARCHIVE_OBJECT();
  3421. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  3422. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Write operations disabled by the php.ini setting phar.readonly");
  3423. return;
  3424. }
  3425. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &metadata) == FAILURE) {
  3426. return;
  3427. }
  3428. if (phar_obj->arc.archive->is_persistent && FAILURE == phar_copy_on_write(&(phar_obj->arc.archive) TSRMLS_CC)) {
  3429. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar_obj->arc.archive->fname);
  3430. return;
  3431. }
  3432. if (phar_obj->arc.archive->metadata) {
  3433. zval_ptr_dtor(&phar_obj->arc.archive->metadata);
  3434. phar_obj->arc.archive->metadata = NULL;
  3435. }
  3436. MAKE_STD_ZVAL(phar_obj->arc.archive->metadata);
  3437. ZVAL_ZVAL(phar_obj->arc.archive->metadata, metadata, 1, 0);
  3438. phar_obj->arc.archive->is_modified = 1;
  3439. phar_flush(phar_obj->arc.archive, 0, 0, 0, &error TSRMLS_CC);
  3440. if (error) {
  3441. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  3442. efree(error);
  3443. }
  3444. }
  3445. /* }}} */
  3446. /* {{{ proto int Phar::delMetadata()
  3447. * Deletes the global metadata of the phar
  3448. */
  3449. PHP_METHOD(Phar, delMetadata)
  3450. {
  3451. char *error;
  3452. PHAR_ARCHIVE_OBJECT();
  3453. if (PHAR_G(readonly) && !phar_obj->arc.archive->is_data) {
  3454. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Write operations disabled by the php.ini setting phar.readonly");
  3455. return;
  3456. }
  3457. if (phar_obj->arc.archive->metadata) {
  3458. zval_ptr_dtor(&phar_obj->arc.archive->metadata);
  3459. phar_obj->arc.archive->metadata = NULL;
  3460. phar_obj->arc.archive->is_modified = 1;
  3461. phar_flush(phar_obj->arc.archive, 0, 0, 0, &error TSRMLS_CC);
  3462. if (error) {
  3463. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  3464. efree(error);
  3465. RETURN_FALSE;
  3466. } else {
  3467. RETURN_TRUE;
  3468. }
  3469. } else {
  3470. RETURN_TRUE;
  3471. }
  3472. }
  3473. /* }}} */
  3474. #if PHP_API_VERSION < 20100412
  3475. #define PHAR_OPENBASEDIR_CHECKPATH(filename) \
  3476. (PG(safe_mode) && (!php_checkuid(filename, NULL, CHECKUID_CHECK_FILE_AND_DIR))) || php_check_open_basedir(filename TSRMLS_CC)
  3477. #else
  3478. #define PHAR_OPENBASEDIR_CHECKPATH(filename) \
  3479. php_check_open_basedir(filename TSRMLS_CC)
  3480. #endif
  3481. static int phar_extract_file(zend_bool overwrite, phar_entry_info *entry, char *dest, int dest_len, char **error TSRMLS_DC) /* {{{ */
  3482. {
  3483. php_stream_statbuf ssb;
  3484. int len;
  3485. php_stream *fp;
  3486. char *fullpath;
  3487. const char *slash;
  3488. mode_t mode;
  3489. cwd_state new_state;
  3490. char *filename;
  3491. size_t filename_len;
  3492. if (entry->is_mounted) {
  3493. /* silently ignore mounted entries */
  3494. return SUCCESS;
  3495. }
  3496. if (entry->filename_len >= sizeof(".phar")-1 && !memcmp(entry->filename, ".phar", sizeof(".phar")-1)) {
  3497. return SUCCESS;
  3498. }
  3499. /* strip .. from path and restrict it to be under dest directory */
  3500. new_state.cwd = (char*)emalloc(2);
  3501. new_state.cwd[0] = DEFAULT_SLASH;
  3502. new_state.cwd[1] = '\0';
  3503. new_state.cwd_length = 1;
  3504. if (virtual_file_ex(&new_state, entry->filename, NULL, CWD_EXPAND TSRMLS_CC) != 0 ||
  3505. new_state.cwd_length <= 1) {
  3506. if (EINVAL == errno && entry->filename_len > 50) {
  3507. char *tmp = estrndup(entry->filename, 50);
  3508. spprintf(error, 4096, "Cannot extract \"%s...\" to \"%s...\", extracted filename is too long for filesystem", tmp, dest);
  3509. efree(tmp);
  3510. } else {
  3511. spprintf(error, 4096, "Cannot extract \"%s\", internal error", entry->filename);
  3512. }
  3513. efree(new_state.cwd);
  3514. return FAILURE;
  3515. }
  3516. filename = new_state.cwd + 1;
  3517. filename_len = new_state.cwd_length - 1;
  3518. #ifdef PHP_WIN32
  3519. /* unixify the path back, otherwise non zip formats might be broken */
  3520. {
  3521. int cnt = filename_len;
  3522. do {
  3523. if ('\\' == filename[cnt]) {
  3524. filename[cnt] = '/';
  3525. }
  3526. } while (cnt-- >= 0);
  3527. }
  3528. #endif
  3529. len = spprintf(&fullpath, 0, "%s/%s", dest, filename);
  3530. if (len >= MAXPATHLEN) {
  3531. char *tmp;
  3532. /* truncate for error message */
  3533. fullpath[50] = '\0';
  3534. if (entry->filename_len > 50) {
  3535. tmp = estrndup(entry->filename, 50);
  3536. spprintf(error, 4096, "Cannot extract \"%s...\" to \"%s...\", extracted filename is too long for filesystem", tmp, fullpath);
  3537. efree(tmp);
  3538. } else {
  3539. spprintf(error, 4096, "Cannot extract \"%s\" to \"%s...\", extracted filename is too long for filesystem", entry->filename, fullpath);
  3540. }
  3541. efree(fullpath);
  3542. efree(new_state.cwd);
  3543. return FAILURE;
  3544. }
  3545. if (!len) {
  3546. spprintf(error, 4096, "Cannot extract \"%s\", internal error", entry->filename);
  3547. efree(fullpath);
  3548. efree(new_state.cwd);
  3549. return FAILURE;
  3550. }
  3551. if (PHAR_OPENBASEDIR_CHECKPATH(fullpath)) {
  3552. spprintf(error, 4096, "Cannot extract \"%s\" to \"%s\", openbasedir/safe mode restrictions in effect", entry->filename, fullpath);
  3553. efree(fullpath);
  3554. efree(new_state.cwd);
  3555. return FAILURE;
  3556. }
  3557. /* let see if the path already exists */
  3558. if (!overwrite && SUCCESS == php_stream_stat_path(fullpath, &ssb)) {
  3559. spprintf(error, 4096, "Cannot extract \"%s\" to \"%s\", path already exists", entry->filename, fullpath);
  3560. efree(fullpath);
  3561. efree(new_state.cwd);
  3562. return FAILURE;
  3563. }
  3564. /* perform dirname */
  3565. slash = zend_memrchr(filename, '/', filename_len);
  3566. if (slash) {
  3567. fullpath[dest_len + (slash - filename) + 1] = '\0';
  3568. } else {
  3569. fullpath[dest_len] = '\0';
  3570. }
  3571. if (FAILURE == php_stream_stat_path(fullpath, &ssb)) {
  3572. if (entry->is_dir) {
  3573. if (!php_stream_mkdir(fullpath, entry->flags & PHAR_ENT_PERM_MASK, PHP_STREAM_MKDIR_RECURSIVE, NULL)) {
  3574. spprintf(error, 4096, "Cannot extract \"%s\", could not create directory \"%s\"", entry->filename, fullpath);
  3575. efree(fullpath);
  3576. efree(new_state.cwd);
  3577. return FAILURE;
  3578. }
  3579. } else {
  3580. if (!php_stream_mkdir(fullpath, 0777, PHP_STREAM_MKDIR_RECURSIVE, NULL)) {
  3581. spprintf(error, 4096, "Cannot extract \"%s\", could not create directory \"%s\"", entry->filename, fullpath);
  3582. efree(fullpath);
  3583. efree(new_state.cwd);
  3584. return FAILURE;
  3585. }
  3586. }
  3587. }
  3588. if (slash) {
  3589. fullpath[dest_len + (slash - filename) + 1] = '/';
  3590. } else {
  3591. fullpath[dest_len] = '/';
  3592. }
  3593. filename = NULL;
  3594. efree(new_state.cwd);
  3595. /* it is a standalone directory, job done */
  3596. if (entry->is_dir) {
  3597. efree(fullpath);
  3598. return SUCCESS;
  3599. }
  3600. #if PHP_API_VERSION < 20100412
  3601. fp = php_stream_open_wrapper(fullpath, "w+b", REPORT_ERRORS|ENFORCE_SAFE_MODE, NULL);
  3602. #else
  3603. fp = php_stream_open_wrapper(fullpath, "w+b", REPORT_ERRORS, NULL);
  3604. #endif
  3605. if (!fp) {
  3606. spprintf(error, 4096, "Cannot extract \"%s\", could not open for writing \"%s\"", entry->filename, fullpath);
  3607. efree(fullpath);
  3608. return FAILURE;
  3609. }
  3610. if (!phar_get_efp(entry, 0 TSRMLS_CC)) {
  3611. if (FAILURE == phar_open_entry_fp(entry, error, 1 TSRMLS_CC)) {
  3612. if (error) {
  3613. spprintf(error, 4096, "Cannot extract \"%s\" to \"%s\", unable to open internal file pointer: %s", entry->filename, fullpath, *error);
  3614. } else {
  3615. spprintf(error, 4096, "Cannot extract \"%s\" to \"%s\", unable to open internal file pointer", entry->filename, fullpath);
  3616. }
  3617. efree(fullpath);
  3618. php_stream_close(fp);
  3619. return FAILURE;
  3620. }
  3621. }
  3622. if (FAILURE == phar_seek_efp(entry, 0, SEEK_SET, 0, 0 TSRMLS_CC)) {
  3623. spprintf(error, 4096, "Cannot extract \"%s\" to \"%s\", unable to seek internal file pointer", entry->filename, fullpath);
  3624. efree(fullpath);
  3625. php_stream_close(fp);
  3626. return FAILURE;
  3627. }
  3628. if (SUCCESS != php_stream_copy_to_stream_ex(phar_get_efp(entry, 0 TSRMLS_CC), fp, entry->uncompressed_filesize, NULL)) {
  3629. spprintf(error, 4096, "Cannot extract \"%s\" to \"%s\", copying contents failed", entry->filename, fullpath);
  3630. efree(fullpath);
  3631. php_stream_close(fp);
  3632. return FAILURE;
  3633. }
  3634. php_stream_close(fp);
  3635. mode = (mode_t) entry->flags & PHAR_ENT_PERM_MASK;
  3636. if (FAILURE == VCWD_CHMOD(fullpath, mode)) {
  3637. spprintf(error, 4096, "Cannot extract \"%s\" to \"%s\", setting file permissions failed", entry->filename, fullpath);
  3638. efree(fullpath);
  3639. return FAILURE;
  3640. }
  3641. efree(fullpath);
  3642. return SUCCESS;
  3643. }
  3644. /* }}} */
  3645. /* {{{ proto bool Phar::extractTo(string pathto[[, mixed files], bool overwrite])
  3646. * Extract one or more file from a phar archive, optionally overwriting existing files
  3647. */
  3648. PHP_METHOD(Phar, extractTo)
  3649. {
  3650. char *error = NULL;
  3651. php_stream *fp;
  3652. php_stream_statbuf ssb;
  3653. phar_entry_info *entry;
  3654. char *pathto, *filename, *actual;
  3655. int pathto_len, filename_len;
  3656. int ret, i;
  3657. int nelems;
  3658. zval *zval_files = NULL;
  3659. zend_bool overwrite = 0;
  3660. PHAR_ARCHIVE_OBJECT();
  3661. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p|z!b", &pathto, &pathto_len, &zval_files, &overwrite) == FAILURE) {
  3662. return;
  3663. }
  3664. fp = php_stream_open_wrapper(phar_obj->arc.archive->fname, "rb", IGNORE_URL|STREAM_MUST_SEEK, &actual);
  3665. if (!fp) {
  3666. zend_throw_exception_ex(spl_ce_InvalidArgumentException, 0 TSRMLS_CC,
  3667. "Invalid argument, %s cannot be found", phar_obj->arc.archive->fname);
  3668. return;
  3669. }
  3670. efree(actual);
  3671. php_stream_close(fp);
  3672. if (pathto_len < 1) {
  3673. zend_throw_exception_ex(spl_ce_InvalidArgumentException, 0 TSRMLS_CC,
  3674. "Invalid argument, extraction path must be non-zero length");
  3675. return;
  3676. }
  3677. if (pathto_len >= MAXPATHLEN) {
  3678. char *tmp = estrndup(pathto, 50);
  3679. /* truncate for error message */
  3680. zend_throw_exception_ex(spl_ce_InvalidArgumentException, 0 TSRMLS_CC, "Cannot extract to \"%s...\", destination directory is too long for filesystem", tmp);
  3681. efree(tmp);
  3682. return;
  3683. }
  3684. if (php_stream_stat_path(pathto, &ssb) < 0) {
  3685. ret = php_stream_mkdir(pathto, 0777, PHP_STREAM_MKDIR_RECURSIVE, NULL);
  3686. if (!ret) {
  3687. zend_throw_exception_ex(spl_ce_RuntimeException, 0 TSRMLS_CC,
  3688. "Unable to create path \"%s\" for extraction", pathto);
  3689. return;
  3690. }
  3691. } else if (!(ssb.sb.st_mode & S_IFDIR)) {
  3692. zend_throw_exception_ex(spl_ce_RuntimeException, 0 TSRMLS_CC,
  3693. "Unable to use path \"%s\" for extraction, it is a file, must be a directory", pathto);
  3694. return;
  3695. }
  3696. if (zval_files) {
  3697. switch (Z_TYPE_P(zval_files)) {
  3698. case IS_NULL:
  3699. goto all_files;
  3700. case IS_STRING:
  3701. filename = Z_STRVAL_P(zval_files);
  3702. filename_len = Z_STRLEN_P(zval_files);
  3703. break;
  3704. case IS_ARRAY:
  3705. nelems = zend_hash_num_elements(Z_ARRVAL_P(zval_files));
  3706. if (nelems == 0 ) {
  3707. RETURN_FALSE;
  3708. }
  3709. for (i = 0; i < nelems; i++) {
  3710. zval **zval_file;
  3711. if (zend_hash_index_find(Z_ARRVAL_P(zval_files), i, (void **) &zval_file) == SUCCESS) {
  3712. switch (Z_TYPE_PP(zval_file)) {
  3713. case IS_STRING:
  3714. break;
  3715. default:
  3716. zend_throw_exception_ex(spl_ce_InvalidArgumentException, 0 TSRMLS_CC,
  3717. "Invalid argument, array of filenames to extract contains non-string value");
  3718. return;
  3719. }
  3720. if (FAILURE == zend_hash_find(&phar_obj->arc.archive->manifest, Z_STRVAL_PP(zval_file), Z_STRLEN_PP(zval_file), (void **)&entry)) {
  3721. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC,
  3722. "Phar Error: attempted to extract non-existent file \"%s\" from phar \"%s\"", Z_STRVAL_PP(zval_file), phar_obj->arc.archive->fname);
  3723. }
  3724. if (FAILURE == phar_extract_file(overwrite, entry, pathto, pathto_len, &error TSRMLS_CC)) {
  3725. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC,
  3726. "Extraction from phar \"%s\" failed: %s", phar_obj->arc.archive->fname, error);
  3727. efree(error);
  3728. return;
  3729. }
  3730. }
  3731. }
  3732. RETURN_TRUE;
  3733. default:
  3734. zend_throw_exception_ex(spl_ce_InvalidArgumentException, 0 TSRMLS_CC,
  3735. "Invalid argument, expected a filename (string) or array of filenames");
  3736. return;
  3737. }
  3738. if (FAILURE == zend_hash_find(&phar_obj->arc.archive->manifest, filename, filename_len, (void **)&entry)) {
  3739. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC,
  3740. "Phar Error: attempted to extract non-existent file \"%s\" from phar \"%s\"", filename, phar_obj->arc.archive->fname);
  3741. return;
  3742. }
  3743. if (FAILURE == phar_extract_file(overwrite, entry, pathto, pathto_len, &error TSRMLS_CC)) {
  3744. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC,
  3745. "Extraction from phar \"%s\" failed: %s", phar_obj->arc.archive->fname, error);
  3746. efree(error);
  3747. return;
  3748. }
  3749. } else {
  3750. phar_archive_data *phar;
  3751. all_files:
  3752. phar = phar_obj->arc.archive;
  3753. /* Extract all files */
  3754. if (!zend_hash_num_elements(&(phar->manifest))) {
  3755. RETURN_TRUE;
  3756. }
  3757. for (zend_hash_internal_pointer_reset(&phar->manifest);
  3758. zend_hash_has_more_elements(&phar->manifest) == SUCCESS;
  3759. zend_hash_move_forward(&phar->manifest)) {
  3760. if (zend_hash_get_current_data(&phar->manifest, (void **)&entry) == FAILURE) {
  3761. continue;
  3762. }
  3763. if (FAILURE == phar_extract_file(overwrite, entry, pathto, pathto_len, &error TSRMLS_CC)) {
  3764. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC,
  3765. "Extraction from phar \"%s\" failed: %s", phar->fname, error);
  3766. efree(error);
  3767. return;
  3768. }
  3769. }
  3770. }
  3771. RETURN_TRUE;
  3772. }
  3773. /* }}} */
  3774. /* {{{ proto void PharFileInfo::__construct(string entry)
  3775. * Construct a Phar entry object
  3776. */
  3777. PHP_METHOD(PharFileInfo, __construct)
  3778. {
  3779. char *fname, *arch, *entry, *error;
  3780. int fname_len, arch_len, entry_len;
  3781. phar_entry_object *entry_obj;
  3782. phar_entry_info *entry_info;
  3783. phar_archive_data *phar_data;
  3784. zval *zobj = getThis(), arg1;
  3785. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p", &fname, &fname_len) == FAILURE) {
  3786. return;
  3787. }
  3788. entry_obj = (phar_entry_object*)zend_object_store_get_object(getThis() TSRMLS_CC);
  3789. if (entry_obj->ent.entry) {
  3790. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Cannot call constructor twice");
  3791. return;
  3792. }
  3793. if (fname_len < 7 || memcmp(fname, "phar://", 7) || phar_split_fname(fname, fname_len, &arch, &arch_len, &entry, &entry_len, 2, 0 TSRMLS_CC) == FAILURE) {
  3794. zend_throw_exception_ex(spl_ce_RuntimeException, 0 TSRMLS_CC,
  3795. "'%s' is not a valid phar archive URL (must have at least phar://filename.phar)", fname);
  3796. return;
  3797. }
  3798. if (phar_open_from_filename(arch, arch_len, NULL, 0, REPORT_ERRORS, &phar_data, &error TSRMLS_CC) == FAILURE) {
  3799. efree(arch);
  3800. efree(entry);
  3801. if (error) {
  3802. zend_throw_exception_ex(spl_ce_RuntimeException, 0 TSRMLS_CC,
  3803. "Cannot open phar file '%s': %s", fname, error);
  3804. efree(error);
  3805. } else {
  3806. zend_throw_exception_ex(spl_ce_RuntimeException, 0 TSRMLS_CC,
  3807. "Cannot open phar file '%s'", fname);
  3808. }
  3809. return;
  3810. }
  3811. if ((entry_info = phar_get_entry_info_dir(phar_data, entry, entry_len, 1, &error, 1 TSRMLS_CC)) == NULL) {
  3812. zend_throw_exception_ex(spl_ce_RuntimeException, 0 TSRMLS_CC,
  3813. "Cannot access phar file entry '%s' in archive '%s'%s%s", entry, arch, error ? ", " : "", error ? error : "");
  3814. efree(arch);
  3815. efree(entry);
  3816. return;
  3817. }
  3818. efree(arch);
  3819. efree(entry);
  3820. entry_obj->ent.entry = entry_info;
  3821. INIT_PZVAL(&arg1);
  3822. ZVAL_STRINGL(&arg1, fname, fname_len, 0);
  3823. zend_call_method_with_1_params(&zobj, Z_OBJCE_P(zobj),
  3824. &spl_ce_SplFileInfo->constructor, "__construct", NULL, &arg1);
  3825. }
  3826. /* }}} */
  3827. #define PHAR_ENTRY_OBJECT() \
  3828. phar_entry_object *entry_obj = (phar_entry_object*)zend_object_store_get_object(getThis() TSRMLS_CC); \
  3829. if (!entry_obj->ent.entry) { \
  3830. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, \
  3831. "Cannot call method on an uninitialized PharFileInfo object"); \
  3832. return; \
  3833. }
  3834. /* {{{ proto void PharFileInfo::__destruct()
  3835. * clean up directory-based entry objects
  3836. */
  3837. PHP_METHOD(PharFileInfo, __destruct)
  3838. {
  3839. phar_entry_object *entry_obj = (phar_entry_object*)zend_object_store_get_object(getThis() TSRMLS_CC); \
  3840. if (entry_obj->ent.entry && entry_obj->ent.entry->is_temp_dir) {
  3841. if (entry_obj->ent.entry->filename) {
  3842. efree(entry_obj->ent.entry->filename);
  3843. entry_obj->ent.entry->filename = NULL;
  3844. }
  3845. efree(entry_obj->ent.entry);
  3846. entry_obj->ent.entry = NULL;
  3847. }
  3848. }
  3849. /* }}} */
  3850. /* {{{ proto int PharFileInfo::getCompressedSize()
  3851. * Returns the compressed size
  3852. */
  3853. PHP_METHOD(PharFileInfo, getCompressedSize)
  3854. {
  3855. PHAR_ENTRY_OBJECT();
  3856. if (zend_parse_parameters_none() == FAILURE) {
  3857. return;
  3858. }
  3859. RETURN_LONG(entry_obj->ent.entry->compressed_filesize);
  3860. }
  3861. /* }}} */
  3862. /* {{{ proto bool PharFileInfo::isCompressed([int compression_type])
  3863. * Returns whether the entry is compressed, and whether it is compressed with Phar::GZ or Phar::BZ2 if specified
  3864. */
  3865. PHP_METHOD(PharFileInfo, isCompressed)
  3866. {
  3867. /* a number that is not Phar::GZ or Phar::BZ2 */
  3868. long method = 9021976;
  3869. PHAR_ENTRY_OBJECT();
  3870. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|l", &method) == FAILURE) {
  3871. return;
  3872. }
  3873. switch (method) {
  3874. case 9021976:
  3875. RETURN_BOOL(entry_obj->ent.entry->flags & PHAR_ENT_COMPRESSION_MASK);
  3876. case PHAR_ENT_COMPRESSED_GZ:
  3877. RETURN_BOOL(entry_obj->ent.entry->flags & PHAR_ENT_COMPRESSED_GZ);
  3878. case PHAR_ENT_COMPRESSED_BZ2:
  3879. RETURN_BOOL(entry_obj->ent.entry->flags & PHAR_ENT_COMPRESSED_BZ2);
  3880. default:
  3881. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, \
  3882. "Unknown compression type specified"); \
  3883. }
  3884. }
  3885. /* }}} */
  3886. /* {{{ proto int PharFileInfo::getCRC32()
  3887. * Returns CRC32 code or throws an exception if not CRC checked
  3888. */
  3889. PHP_METHOD(PharFileInfo, getCRC32)
  3890. {
  3891. PHAR_ENTRY_OBJECT();
  3892. if (zend_parse_parameters_none() == FAILURE) {
  3893. return;
  3894. }
  3895. if (entry_obj->ent.entry->is_dir) {
  3896. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, \
  3897. "Phar entry is a directory, does not have a CRC"); \
  3898. return;
  3899. }
  3900. if (entry_obj->ent.entry->is_crc_checked) {
  3901. RETURN_LONG(entry_obj->ent.entry->crc32);
  3902. } else {
  3903. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, \
  3904. "Phar entry was not CRC checked"); \
  3905. }
  3906. }
  3907. /* }}} */
  3908. /* {{{ proto int PharFileInfo::isCRCChecked()
  3909. * Returns whether file entry is CRC checked
  3910. */
  3911. PHP_METHOD(PharFileInfo, isCRCChecked)
  3912. {
  3913. PHAR_ENTRY_OBJECT();
  3914. if (zend_parse_parameters_none() == FAILURE) {
  3915. return;
  3916. }
  3917. RETURN_BOOL(entry_obj->ent.entry->is_crc_checked);
  3918. }
  3919. /* }}} */
  3920. /* {{{ proto int PharFileInfo::getPharFlags()
  3921. * Returns the Phar file entry flags
  3922. */
  3923. PHP_METHOD(PharFileInfo, getPharFlags)
  3924. {
  3925. PHAR_ENTRY_OBJECT();
  3926. if (zend_parse_parameters_none() == FAILURE) {
  3927. return;
  3928. }
  3929. RETURN_LONG(entry_obj->ent.entry->flags & ~(PHAR_ENT_PERM_MASK|PHAR_ENT_COMPRESSION_MASK));
  3930. }
  3931. /* }}} */
  3932. /* {{{ proto int PharFileInfo::chmod()
  3933. * set the file permissions for the Phar. This only allows setting execution bit, read/write
  3934. */
  3935. PHP_METHOD(PharFileInfo, chmod)
  3936. {
  3937. char *error;
  3938. long perms;
  3939. PHAR_ENTRY_OBJECT();
  3940. if (entry_obj->ent.entry->is_temp_dir) {
  3941. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, \
  3942. "Phar entry \"%s\" is a temporary directory (not an actual entry in the archive), cannot chmod", entry_obj->ent.entry->filename); \
  3943. return;
  3944. }
  3945. if (PHAR_G(readonly) && !entry_obj->ent.entry->phar->is_data) {
  3946. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "Cannot modify permissions for file \"%s\" in phar \"%s\", write operations are prohibited", entry_obj->ent.entry->filename, entry_obj->ent.entry->phar->fname);
  3947. return;
  3948. }
  3949. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &perms) == FAILURE) {
  3950. return;
  3951. }
  3952. if (entry_obj->ent.entry->is_persistent) {
  3953. phar_archive_data *phar = entry_obj->ent.entry->phar;
  3954. if (FAILURE == phar_copy_on_write(&phar TSRMLS_CC)) {
  3955. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar->fname);
  3956. return;
  3957. }
  3958. /* re-populate after copy-on-write */
  3959. zend_hash_find(&phar->manifest, entry_obj->ent.entry->filename, entry_obj->ent.entry->filename_len, (void **)&entry_obj->ent.entry);
  3960. }
  3961. /* clear permissions */
  3962. entry_obj->ent.entry->flags &= ~PHAR_ENT_PERM_MASK;
  3963. perms &= 0777;
  3964. entry_obj->ent.entry->flags |= perms;
  3965. entry_obj->ent.entry->old_flags = entry_obj->ent.entry->flags;
  3966. entry_obj->ent.entry->phar->is_modified = 1;
  3967. entry_obj->ent.entry->is_modified = 1;
  3968. /* hackish cache in php_stat needs to be cleared */
  3969. /* if this code fails to work, check main/streams/streams.c, _php_stream_stat_path */
  3970. if (BG(CurrentLStatFile)) {
  3971. efree(BG(CurrentLStatFile));
  3972. }
  3973. if (BG(CurrentStatFile)) {
  3974. efree(BG(CurrentStatFile));
  3975. }
  3976. BG(CurrentLStatFile) = NULL;
  3977. BG(CurrentStatFile) = NULL;
  3978. phar_flush(entry_obj->ent.entry->phar, 0, 0, 0, &error TSRMLS_CC);
  3979. if (error) {
  3980. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  3981. efree(error);
  3982. }
  3983. }
  3984. /* }}} */
  3985. /* {{{ proto int PharFileInfo::hasMetaData()
  3986. * Returns the metadata of the entry
  3987. */
  3988. PHP_METHOD(PharFileInfo, hasMetadata)
  3989. {
  3990. PHAR_ENTRY_OBJECT();
  3991. if (zend_parse_parameters_none() == FAILURE) {
  3992. return;
  3993. }
  3994. RETURN_BOOL(entry_obj->ent.entry->metadata != NULL);
  3995. }
  3996. /* }}} */
  3997. /* {{{ proto int PharFileInfo::getMetaData()
  3998. * Returns the metadata of the entry
  3999. */
  4000. PHP_METHOD(PharFileInfo, getMetadata)
  4001. {
  4002. PHAR_ENTRY_OBJECT();
  4003. if (zend_parse_parameters_none() == FAILURE) {
  4004. return;
  4005. }
  4006. if (entry_obj->ent.entry->metadata) {
  4007. if (entry_obj->ent.entry->is_persistent) {
  4008. zval *ret;
  4009. char *buf = estrndup((char *) entry_obj->ent.entry->metadata, entry_obj->ent.entry->metadata_len);
  4010. /* assume success, we would have failed before */
  4011. phar_parse_metadata(&buf, &ret, entry_obj->ent.entry->metadata_len TSRMLS_CC);
  4012. efree(buf);
  4013. RETURN_ZVAL(ret, 0, 1);
  4014. }
  4015. RETURN_ZVAL(entry_obj->ent.entry->metadata, 1, 0);
  4016. }
  4017. }
  4018. /* }}} */
  4019. /* {{{ proto int PharFileInfo::setMetaData(mixed $metadata)
  4020. * Sets the metadata of the entry
  4021. */
  4022. PHP_METHOD(PharFileInfo, setMetadata)
  4023. {
  4024. char *error;
  4025. zval *metadata;
  4026. PHAR_ENTRY_OBJECT();
  4027. if (PHAR_G(readonly) && !entry_obj->ent.entry->phar->is_data) {
  4028. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Write operations disabled by the php.ini setting phar.readonly");
  4029. return;
  4030. }
  4031. if (entry_obj->ent.entry->is_temp_dir) {
  4032. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, \
  4033. "Phar entry is a temporary directory (not an actual entry in the archive), cannot set metadata"); \
  4034. return;
  4035. }
  4036. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &metadata) == FAILURE) {
  4037. return;
  4038. }
  4039. if (entry_obj->ent.entry->is_persistent) {
  4040. phar_archive_data *phar = entry_obj->ent.entry->phar;
  4041. if (FAILURE == phar_copy_on_write(&phar TSRMLS_CC)) {
  4042. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar->fname);
  4043. return;
  4044. }
  4045. /* re-populate after copy-on-write */
  4046. zend_hash_find(&phar->manifest, entry_obj->ent.entry->filename, entry_obj->ent.entry->filename_len, (void **)&entry_obj->ent.entry);
  4047. }
  4048. if (entry_obj->ent.entry->metadata) {
  4049. zval_ptr_dtor(&entry_obj->ent.entry->metadata);
  4050. entry_obj->ent.entry->metadata = NULL;
  4051. }
  4052. MAKE_STD_ZVAL(entry_obj->ent.entry->metadata);
  4053. ZVAL_ZVAL(entry_obj->ent.entry->metadata, metadata, 1, 0);
  4054. entry_obj->ent.entry->is_modified = 1;
  4055. entry_obj->ent.entry->phar->is_modified = 1;
  4056. phar_flush(entry_obj->ent.entry->phar, 0, 0, 0, &error TSRMLS_CC);
  4057. if (error) {
  4058. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  4059. efree(error);
  4060. }
  4061. }
  4062. /* }}} */
  4063. /* {{{ proto bool PharFileInfo::delMetaData()
  4064. * Deletes the metadata of the entry
  4065. */
  4066. PHP_METHOD(PharFileInfo, delMetadata)
  4067. {
  4068. char *error;
  4069. PHAR_ENTRY_OBJECT();
  4070. if (zend_parse_parameters_none() == FAILURE) {
  4071. return;
  4072. }
  4073. if (PHAR_G(readonly) && !entry_obj->ent.entry->phar->is_data) {
  4074. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Write operations disabled by the php.ini setting phar.readonly");
  4075. return;
  4076. }
  4077. if (entry_obj->ent.entry->is_temp_dir) {
  4078. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, \
  4079. "Phar entry is a temporary directory (not an actual entry in the archive), cannot delete metadata"); \
  4080. return;
  4081. }
  4082. if (entry_obj->ent.entry->metadata) {
  4083. if (entry_obj->ent.entry->is_persistent) {
  4084. phar_archive_data *phar = entry_obj->ent.entry->phar;
  4085. if (FAILURE == phar_copy_on_write(&phar TSRMLS_CC)) {
  4086. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar->fname);
  4087. return;
  4088. }
  4089. /* re-populate after copy-on-write */
  4090. zend_hash_find(&phar->manifest, entry_obj->ent.entry->filename, entry_obj->ent.entry->filename_len, (void **)&entry_obj->ent.entry);
  4091. }
  4092. zval_ptr_dtor(&entry_obj->ent.entry->metadata);
  4093. entry_obj->ent.entry->metadata = NULL;
  4094. entry_obj->ent.entry->is_modified = 1;
  4095. entry_obj->ent.entry->phar->is_modified = 1;
  4096. phar_flush(entry_obj->ent.entry->phar, 0, 0, 0, &error TSRMLS_CC);
  4097. if (error) {
  4098. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  4099. efree(error);
  4100. RETURN_FALSE;
  4101. } else {
  4102. RETURN_TRUE;
  4103. }
  4104. } else {
  4105. RETURN_TRUE;
  4106. }
  4107. }
  4108. /* }}} */
  4109. /* {{{ proto string PharFileInfo::getContent()
  4110. * return the complete file contents of the entry (like file_get_contents)
  4111. */
  4112. PHP_METHOD(PharFileInfo, getContent)
  4113. {
  4114. char *error;
  4115. php_stream *fp;
  4116. phar_entry_info *link;
  4117. PHAR_ENTRY_OBJECT();
  4118. if (zend_parse_parameters_none() == FAILURE) {
  4119. return;
  4120. }
  4121. if (entry_obj->ent.entry->is_dir) {
  4122. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4123. "Phar error: Cannot retrieve contents, \"%s\" in phar \"%s\" is a directory", entry_obj->ent.entry->filename, entry_obj->ent.entry->phar->fname);
  4124. return;
  4125. }
  4126. link = phar_get_link_source(entry_obj->ent.entry TSRMLS_CC);
  4127. if (!link) {
  4128. link = entry_obj->ent.entry;
  4129. }
  4130. if (SUCCESS != phar_open_entry_fp(link, &error, 0 TSRMLS_CC)) {
  4131. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4132. "Phar error: Cannot retrieve contents, \"%s\" in phar \"%s\": %s", entry_obj->ent.entry->filename, entry_obj->ent.entry->phar->fname, error);
  4133. efree(error);
  4134. return;
  4135. }
  4136. if (!(fp = phar_get_efp(link, 0 TSRMLS_CC))) {
  4137. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4138. "Phar error: Cannot retrieve contents of \"%s\" in phar \"%s\"", entry_obj->ent.entry->filename, entry_obj->ent.entry->phar->fname);
  4139. return;
  4140. }
  4141. phar_seek_efp(link, 0, SEEK_SET, 0, 0 TSRMLS_CC);
  4142. Z_TYPE_P(return_value) = IS_STRING;
  4143. Z_STRVAL_P(return_value) = NULL;
  4144. Z_STRLEN_P(return_value) = php_stream_copy_to_mem(fp, &(Z_STRVAL_P(return_value)), link->uncompressed_filesize, 0);
  4145. if (!Z_STRVAL_P(return_value)) {
  4146. Z_STRVAL_P(return_value) = estrndup("", 0);
  4147. }
  4148. }
  4149. /* }}} */
  4150. /* {{{ proto int PharFileInfo::compress(int compression_type)
  4151. * Instructs the Phar class to compress the current file using zlib or bzip2 compression
  4152. */
  4153. PHP_METHOD(PharFileInfo, compress)
  4154. {
  4155. long method;
  4156. char *error;
  4157. PHAR_ENTRY_OBJECT();
  4158. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &method) == FAILURE) {
  4159. return;
  4160. }
  4161. if (entry_obj->ent.entry->is_tar) {
  4162. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4163. "Cannot compress with Gzip compression, not possible with tar-based phar archives");
  4164. return;
  4165. }
  4166. if (entry_obj->ent.entry->is_dir) {
  4167. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, \
  4168. "Phar entry is a directory, cannot set compression"); \
  4169. return;
  4170. }
  4171. if (PHAR_G(readonly) && !entry_obj->ent.entry->phar->is_data) {
  4172. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4173. "Phar is readonly, cannot change compression");
  4174. return;
  4175. }
  4176. if (entry_obj->ent.entry->is_deleted) {
  4177. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4178. "Cannot compress deleted file");
  4179. return;
  4180. }
  4181. if (entry_obj->ent.entry->is_persistent) {
  4182. phar_archive_data *phar = entry_obj->ent.entry->phar;
  4183. if (FAILURE == phar_copy_on_write(&phar TSRMLS_CC)) {
  4184. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar->fname);
  4185. return;
  4186. }
  4187. /* re-populate after copy-on-write */
  4188. zend_hash_find(&phar->manifest, entry_obj->ent.entry->filename, entry_obj->ent.entry->filename_len, (void **)&entry_obj->ent.entry);
  4189. }
  4190. switch (method) {
  4191. case PHAR_ENT_COMPRESSED_GZ:
  4192. if (entry_obj->ent.entry->flags & PHAR_ENT_COMPRESSED_GZ) {
  4193. RETURN_TRUE;
  4194. }
  4195. if ((entry_obj->ent.entry->flags & PHAR_ENT_COMPRESSED_BZ2) != 0) {
  4196. if (!PHAR_G(has_bz2)) {
  4197. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4198. "Cannot compress with gzip compression, file is already compressed with bzip2 compression and bz2 extension is not enabled, cannot decompress");
  4199. return;
  4200. }
  4201. /* decompress this file indirectly */
  4202. if (SUCCESS != phar_open_entry_fp(entry_obj->ent.entry, &error, 1 TSRMLS_CC)) {
  4203. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4204. "Phar error: Cannot decompress bzip2-compressed file \"%s\" in phar \"%s\" in order to compress with gzip: %s", entry_obj->ent.entry->filename, entry_obj->ent.entry->phar->fname, error);
  4205. efree(error);
  4206. return;
  4207. }
  4208. }
  4209. if (!PHAR_G(has_zlib)) {
  4210. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4211. "Cannot compress with gzip compression, zlib extension is not enabled");
  4212. return;
  4213. }
  4214. entry_obj->ent.entry->old_flags = entry_obj->ent.entry->flags;
  4215. entry_obj->ent.entry->flags &= ~PHAR_ENT_COMPRESSION_MASK;
  4216. entry_obj->ent.entry->flags |= PHAR_ENT_COMPRESSED_GZ;
  4217. break;
  4218. case PHAR_ENT_COMPRESSED_BZ2:
  4219. if (entry_obj->ent.entry->flags & PHAR_ENT_COMPRESSED_BZ2) {
  4220. RETURN_TRUE;
  4221. }
  4222. if ((entry_obj->ent.entry->flags & PHAR_ENT_COMPRESSED_GZ) != 0) {
  4223. if (!PHAR_G(has_zlib)) {
  4224. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4225. "Cannot compress with bzip2 compression, file is already compressed with gzip compression and zlib extension is not enabled, cannot decompress");
  4226. return;
  4227. }
  4228. /* decompress this file indirectly */
  4229. if (SUCCESS != phar_open_entry_fp(entry_obj->ent.entry, &error, 1 TSRMLS_CC)) {
  4230. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4231. "Phar error: Cannot decompress gzip-compressed file \"%s\" in phar \"%s\" in order to compress with bzip2: %s", entry_obj->ent.entry->filename, entry_obj->ent.entry->phar->fname, error);
  4232. efree(error);
  4233. return;
  4234. }
  4235. }
  4236. if (!PHAR_G(has_bz2)) {
  4237. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4238. "Cannot compress with bzip2 compression, bz2 extension is not enabled");
  4239. return;
  4240. }
  4241. entry_obj->ent.entry->old_flags = entry_obj->ent.entry->flags;
  4242. entry_obj->ent.entry->flags &= ~PHAR_ENT_COMPRESSION_MASK;
  4243. entry_obj->ent.entry->flags |= PHAR_ENT_COMPRESSED_BZ2;
  4244. break;
  4245. default:
  4246. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, \
  4247. "Unknown compression type specified"); \
  4248. }
  4249. entry_obj->ent.entry->phar->is_modified = 1;
  4250. entry_obj->ent.entry->is_modified = 1;
  4251. phar_flush(entry_obj->ent.entry->phar, 0, 0, 0, &error TSRMLS_CC);
  4252. if (error) {
  4253. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  4254. efree(error);
  4255. }
  4256. RETURN_TRUE;
  4257. }
  4258. /* }}} */
  4259. /* {{{ proto int PharFileInfo::decompress()
  4260. * Instructs the Phar class to decompress the current file
  4261. */
  4262. PHP_METHOD(PharFileInfo, decompress)
  4263. {
  4264. char *error;
  4265. PHAR_ENTRY_OBJECT();
  4266. if (zend_parse_parameters_none() == FAILURE) {
  4267. return;
  4268. }
  4269. if (entry_obj->ent.entry->is_dir) {
  4270. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, \
  4271. "Phar entry is a directory, cannot set compression"); \
  4272. return;
  4273. }
  4274. if ((entry_obj->ent.entry->flags & PHAR_ENT_COMPRESSION_MASK) == 0) {
  4275. RETURN_TRUE;
  4276. }
  4277. if (PHAR_G(readonly) && !entry_obj->ent.entry->phar->is_data) {
  4278. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4279. "Phar is readonly, cannot decompress");
  4280. return;
  4281. }
  4282. if (entry_obj->ent.entry->is_deleted) {
  4283. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4284. "Cannot compress deleted file");
  4285. return;
  4286. }
  4287. if ((entry_obj->ent.entry->flags & PHAR_ENT_COMPRESSED_GZ) != 0 && !PHAR_G(has_zlib)) {
  4288. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4289. "Cannot decompress Gzip-compressed file, zlib extension is not enabled");
  4290. return;
  4291. }
  4292. if ((entry_obj->ent.entry->flags & PHAR_ENT_COMPRESSED_BZ2) != 0 && !PHAR_G(has_bz2)) {
  4293. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC,
  4294. "Cannot decompress Bzip2-compressed file, bz2 extension is not enabled");
  4295. return;
  4296. }
  4297. if (entry_obj->ent.entry->is_persistent) {
  4298. phar_archive_data *phar = entry_obj->ent.entry->phar;
  4299. if (FAILURE == phar_copy_on_write(&phar TSRMLS_CC)) {
  4300. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "phar \"%s\" is persistent, unable to copy on write", phar->fname);
  4301. return;
  4302. }
  4303. /* re-populate after copy-on-write */
  4304. zend_hash_find(&phar->manifest, entry_obj->ent.entry->filename, entry_obj->ent.entry->filename_len, (void **)&entry_obj->ent.entry);
  4305. }
  4306. if (!entry_obj->ent.entry->fp) {
  4307. if (FAILURE == phar_open_archive_fp(entry_obj->ent.entry->phar TSRMLS_CC)) {
  4308. zend_throw_exception_ex(spl_ce_BadMethodCallException, 0 TSRMLS_CC, "Cannot decompress entry \"%s\", phar error: Cannot open phar archive \"%s\" for reading", entry_obj->ent.entry->filename, entry_obj->ent.entry->phar->fname);
  4309. return;
  4310. }
  4311. entry_obj->ent.entry->fp_type = PHAR_FP;
  4312. }
  4313. entry_obj->ent.entry->old_flags = entry_obj->ent.entry->flags;
  4314. entry_obj->ent.entry->flags &= ~PHAR_ENT_COMPRESSION_MASK;
  4315. entry_obj->ent.entry->phar->is_modified = 1;
  4316. entry_obj->ent.entry->is_modified = 1;
  4317. phar_flush(entry_obj->ent.entry->phar, 0, 0, 0, &error TSRMLS_CC);
  4318. if (error) {
  4319. zend_throw_exception_ex(phar_ce_PharException, 0 TSRMLS_CC, "%s", error);
  4320. efree(error);
  4321. }
  4322. RETURN_TRUE;
  4323. }
  4324. /* }}} */
  4325. #endif /* HAVE_SPL */
  4326. /* {{{ phar methods */
  4327. PHAR_ARG_INFO
  4328. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar___construct, 0, 0, 1)
  4329. ZEND_ARG_INFO(0, filename)
  4330. ZEND_ARG_INFO(0, flags)
  4331. ZEND_ARG_INFO(0, alias)
  4332. ZEND_ARG_INFO(0, fileformat)
  4333. ZEND_END_ARG_INFO()
  4334. PHAR_ARG_INFO
  4335. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_createDS, 0, 0, 0)
  4336. ZEND_ARG_INFO(0, index)
  4337. ZEND_ARG_INFO(0, webindex)
  4338. ZEND_END_ARG_INFO()
  4339. PHAR_ARG_INFO
  4340. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_cancompress, 0, 0, 0)
  4341. ZEND_ARG_INFO(0, method)
  4342. ZEND_END_ARG_INFO()
  4343. PHAR_ARG_INFO
  4344. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_isvalidpharfilename, 0, 0, 1)
  4345. ZEND_ARG_INFO(0, filename)
  4346. ZEND_ARG_INFO(0, executable)
  4347. ZEND_END_ARG_INFO()
  4348. PHAR_ARG_INFO
  4349. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_loadPhar, 0, 0, 1)
  4350. ZEND_ARG_INFO(0, filename)
  4351. ZEND_ARG_INFO(0, alias)
  4352. ZEND_END_ARG_INFO()
  4353. PHAR_ARG_INFO
  4354. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_mapPhar, 0, 0, 0)
  4355. ZEND_ARG_INFO(0, alias)
  4356. ZEND_ARG_INFO(0, offset)
  4357. ZEND_END_ARG_INFO()
  4358. PHAR_ARG_INFO
  4359. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_mount, 0, 0, 2)
  4360. ZEND_ARG_INFO(0, inphar)
  4361. ZEND_ARG_INFO(0, externalfile)
  4362. ZEND_END_ARG_INFO()
  4363. PHAR_ARG_INFO
  4364. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_mungServer, 0, 0, 1)
  4365. ZEND_ARG_INFO(0, munglist)
  4366. ZEND_END_ARG_INFO()
  4367. PHAR_ARG_INFO
  4368. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_webPhar, 0, 0, 0)
  4369. ZEND_ARG_INFO(0, alias)
  4370. ZEND_ARG_INFO(0, index)
  4371. ZEND_ARG_INFO(0, f404)
  4372. ZEND_ARG_INFO(0, mimetypes)
  4373. ZEND_ARG_INFO(0, rewrites)
  4374. ZEND_END_ARG_INFO()
  4375. PHAR_ARG_INFO
  4376. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_running, 0, 0, 1)
  4377. ZEND_ARG_INFO(0, retphar)
  4378. ZEND_END_ARG_INFO()
  4379. PHAR_ARG_INFO
  4380. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_ua, 0, 0, 1)
  4381. ZEND_ARG_INFO(0, archive)
  4382. ZEND_END_ARG_INFO()
  4383. #if HAVE_SPL
  4384. PHAR_ARG_INFO
  4385. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_build, 0, 0, 1)
  4386. ZEND_ARG_INFO(0, iterator)
  4387. ZEND_ARG_INFO(0, base_directory)
  4388. ZEND_END_ARG_INFO()
  4389. PHAR_ARG_INFO
  4390. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_conv, 0, 0, 0)
  4391. ZEND_ARG_INFO(0, format)
  4392. ZEND_ARG_INFO(0, compression_type)
  4393. ZEND_ARG_INFO(0, file_ext)
  4394. ZEND_END_ARG_INFO()
  4395. PHAR_ARG_INFO
  4396. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_comps, 0, 0, 1)
  4397. ZEND_ARG_INFO(0, compression_type)
  4398. ZEND_ARG_INFO(0, file_ext)
  4399. ZEND_END_ARG_INFO()
  4400. PHAR_ARG_INFO
  4401. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_decomp, 0, 0, 0)
  4402. ZEND_ARG_INFO(0, file_ext)
  4403. ZEND_END_ARG_INFO()
  4404. PHAR_ARG_INFO
  4405. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_comp, 0, 0, 1)
  4406. ZEND_ARG_INFO(0, compression_type)
  4407. ZEND_END_ARG_INFO()
  4408. PHAR_ARG_INFO
  4409. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_compo, 0, 0, 0)
  4410. ZEND_ARG_INFO(0, compression_type)
  4411. ZEND_END_ARG_INFO()
  4412. PHAR_ARG_INFO
  4413. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_copy, 0, 0, 2)
  4414. ZEND_ARG_INFO(0, newfile)
  4415. ZEND_ARG_INFO(0, oldfile)
  4416. ZEND_END_ARG_INFO()
  4417. PHAR_ARG_INFO
  4418. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_delete, 0, 0, 1)
  4419. ZEND_ARG_INFO(0, entry)
  4420. ZEND_END_ARG_INFO()
  4421. PHAR_ARG_INFO
  4422. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_fromdir, 0, 0, 1)
  4423. ZEND_ARG_INFO(0, base_dir)
  4424. ZEND_ARG_INFO(0, regex)
  4425. ZEND_END_ARG_INFO()
  4426. PHAR_ARG_INFO
  4427. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_offsetExists, 0, 0, 1)
  4428. ZEND_ARG_INFO(0, entry)
  4429. ZEND_END_ARG_INFO()
  4430. PHAR_ARG_INFO
  4431. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_offsetSet, 0, 0, 2)
  4432. ZEND_ARG_INFO(0, entry)
  4433. ZEND_ARG_INFO(0, value)
  4434. ZEND_END_ARG_INFO()
  4435. PHAR_ARG_INFO
  4436. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_setAlias, 0, 0, 1)
  4437. ZEND_ARG_INFO(0, alias)
  4438. ZEND_END_ARG_INFO()
  4439. PHAR_ARG_INFO
  4440. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_setMetadata, 0, 0, 1)
  4441. ZEND_ARG_INFO(0, metadata)
  4442. ZEND_END_ARG_INFO()
  4443. PHAR_ARG_INFO
  4444. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_setSigAlgo, 0, 0, 1)
  4445. ZEND_ARG_INFO(0, algorithm)
  4446. ZEND_ARG_INFO(0, privatekey)
  4447. ZEND_END_ARG_INFO()
  4448. PHAR_ARG_INFO
  4449. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_setStub, 0, 0, 1)
  4450. ZEND_ARG_INFO(0, newstub)
  4451. ZEND_ARG_INFO(0, maxlen)
  4452. ZEND_END_ARG_INFO()
  4453. PHAR_ARG_INFO
  4454. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_emptydir, 0, 0, 0)
  4455. ZEND_ARG_INFO(0, dirname)
  4456. ZEND_END_ARG_INFO()
  4457. PHAR_ARG_INFO
  4458. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_extract, 0, 0, 1)
  4459. ZEND_ARG_INFO(0, pathto)
  4460. ZEND_ARG_INFO(0, files)
  4461. ZEND_ARG_INFO(0, overwrite)
  4462. ZEND_END_ARG_INFO()
  4463. PHAR_ARG_INFO
  4464. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_addfile, 0, 0, 1)
  4465. ZEND_ARG_INFO(0, filename)
  4466. ZEND_ARG_INFO(0, localname)
  4467. ZEND_END_ARG_INFO()
  4468. PHAR_ARG_INFO
  4469. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_fromstring, 0, 0, 1)
  4470. ZEND_ARG_INFO(0, localname)
  4471. ZEND_ARG_INFO(0, contents)
  4472. ZEND_END_ARG_INFO()
  4473. PHAR_ARG_INFO
  4474. ZEND_BEGIN_ARG_INFO_EX(arginfo_phar_isff, 0, 0, 1)
  4475. ZEND_ARG_INFO(0, fileformat)
  4476. ZEND_END_ARG_INFO()
  4477. PHAR_ARG_INFO
  4478. ZEND_BEGIN_ARG_INFO(arginfo_phar__void, 0)
  4479. ZEND_END_ARG_INFO()
  4480. #endif /* HAVE_SPL */
  4481. zend_function_entry php_archive_methods[] = {
  4482. #if !HAVE_SPL
  4483. PHP_ME(Phar, __construct, arginfo_phar___construct, ZEND_ACC_PRIVATE)
  4484. #else
  4485. PHP_ME(Phar, __construct, arginfo_phar___construct, ZEND_ACC_PUBLIC)
  4486. PHP_ME(Phar, __destruct, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4487. PHP_ME(Phar, addEmptyDir, arginfo_phar_emptydir, ZEND_ACC_PUBLIC)
  4488. PHP_ME(Phar, addFile, arginfo_phar_addfile, ZEND_ACC_PUBLIC)
  4489. PHP_ME(Phar, addFromString, arginfo_phar_fromstring, ZEND_ACC_PUBLIC)
  4490. PHP_ME(Phar, buildFromDirectory, arginfo_phar_fromdir, ZEND_ACC_PUBLIC)
  4491. PHP_ME(Phar, buildFromIterator, arginfo_phar_build, ZEND_ACC_PUBLIC)
  4492. PHP_ME(Phar, compressFiles, arginfo_phar_comp, ZEND_ACC_PUBLIC)
  4493. PHP_ME(Phar, decompressFiles, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4494. PHP_ME(Phar, compress, arginfo_phar_comps, ZEND_ACC_PUBLIC)
  4495. PHP_ME(Phar, decompress, arginfo_phar_decomp, ZEND_ACC_PUBLIC)
  4496. PHP_ME(Phar, convertToExecutable, arginfo_phar_conv, ZEND_ACC_PUBLIC)
  4497. PHP_ME(Phar, convertToData, arginfo_phar_conv, ZEND_ACC_PUBLIC)
  4498. PHP_ME(Phar, copy, arginfo_phar_copy, ZEND_ACC_PUBLIC)
  4499. PHP_ME(Phar, count, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4500. PHP_ME(Phar, delete, arginfo_phar_delete, ZEND_ACC_PUBLIC)
  4501. PHP_ME(Phar, delMetadata, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4502. PHP_ME(Phar, extractTo, arginfo_phar_extract, ZEND_ACC_PUBLIC)
  4503. PHP_ME(Phar, getAlias, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4504. PHP_ME(Phar, getPath, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4505. PHP_ME(Phar, getMetadata, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4506. PHP_ME(Phar, getModified, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4507. PHP_ME(Phar, getSignature, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4508. PHP_ME(Phar, getStub, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4509. PHP_ME(Phar, getVersion, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4510. PHP_ME(Phar, hasMetadata, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4511. PHP_ME(Phar, isBuffering, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4512. PHP_ME(Phar, isCompressed, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4513. PHP_ME(Phar, isFileFormat, arginfo_phar_isff, ZEND_ACC_PUBLIC)
  4514. PHP_ME(Phar, isWritable, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4515. PHP_ME(Phar, offsetExists, arginfo_phar_offsetExists, ZEND_ACC_PUBLIC)
  4516. PHP_ME(Phar, offsetGet, arginfo_phar_offsetExists, ZEND_ACC_PUBLIC)
  4517. PHP_ME(Phar, offsetSet, arginfo_phar_offsetSet, ZEND_ACC_PUBLIC)
  4518. PHP_ME(Phar, offsetUnset, arginfo_phar_offsetExists, ZEND_ACC_PUBLIC)
  4519. PHP_ME(Phar, setAlias, arginfo_phar_setAlias, ZEND_ACC_PUBLIC)
  4520. PHP_ME(Phar, setDefaultStub, arginfo_phar_createDS, ZEND_ACC_PUBLIC)
  4521. PHP_ME(Phar, setMetadata, arginfo_phar_setMetadata, ZEND_ACC_PUBLIC)
  4522. PHP_ME(Phar, setSignatureAlgorithm, arginfo_phar_setSigAlgo, ZEND_ACC_PUBLIC)
  4523. PHP_ME(Phar, setStub, arginfo_phar_setStub, ZEND_ACC_PUBLIC)
  4524. PHP_ME(Phar, startBuffering, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4525. PHP_ME(Phar, stopBuffering, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4526. #endif
  4527. /* static member functions */
  4528. PHP_ME(Phar, apiVersion, arginfo_phar__void, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC|ZEND_ACC_FINAL)
  4529. PHP_ME(Phar, canCompress, arginfo_phar_cancompress, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC|ZEND_ACC_FINAL)
  4530. PHP_ME(Phar, canWrite, arginfo_phar__void, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC|ZEND_ACC_FINAL)
  4531. PHP_ME(Phar, createDefaultStub, arginfo_phar_createDS, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC|ZEND_ACC_FINAL)
  4532. PHP_ME(Phar, getSupportedCompression,arginfo_phar__void, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC|ZEND_ACC_FINAL)
  4533. PHP_ME(Phar, getSupportedSignatures,arginfo_phar__void, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC|ZEND_ACC_FINAL)
  4534. PHP_ME(Phar, interceptFileFuncs, arginfo_phar__void, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC|ZEND_ACC_FINAL)
  4535. PHP_ME(Phar, isValidPharFilename, arginfo_phar_isvalidpharfilename, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC|ZEND_ACC_FINAL)
  4536. PHP_ME(Phar, loadPhar, arginfo_phar_loadPhar, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC|ZEND_ACC_FINAL)
  4537. PHP_ME(Phar, mapPhar, arginfo_phar_mapPhar, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC|ZEND_ACC_FINAL)
  4538. PHP_ME(Phar, running, arginfo_phar_running, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC|ZEND_ACC_FINAL)
  4539. PHP_ME(Phar, mount, arginfo_phar_mount, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC|ZEND_ACC_FINAL)
  4540. PHP_ME(Phar, mungServer, arginfo_phar_mungServer, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC|ZEND_ACC_FINAL)
  4541. PHP_ME(Phar, unlinkArchive, arginfo_phar_ua, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC|ZEND_ACC_FINAL)
  4542. PHP_ME(Phar, webPhar, arginfo_phar_webPhar, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC|ZEND_ACC_FINAL)
  4543. PHP_FE_END
  4544. };
  4545. #if HAVE_SPL
  4546. PHAR_ARG_INFO
  4547. ZEND_BEGIN_ARG_INFO_EX(arginfo_entry___construct, 0, 0, 1)
  4548. ZEND_ARG_INFO(0, filename)
  4549. ZEND_END_ARG_INFO()
  4550. PHAR_ARG_INFO
  4551. ZEND_BEGIN_ARG_INFO_EX(arginfo_entry_chmod, 0, 0, 1)
  4552. ZEND_ARG_INFO(0, perms)
  4553. ZEND_END_ARG_INFO()
  4554. zend_function_entry php_entry_methods[] = {
  4555. PHP_ME(PharFileInfo, __construct, arginfo_entry___construct, ZEND_ACC_PUBLIC)
  4556. PHP_ME(PharFileInfo, __destruct, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4557. PHP_ME(PharFileInfo, chmod, arginfo_entry_chmod, ZEND_ACC_PUBLIC)
  4558. PHP_ME(PharFileInfo, compress, arginfo_phar_comp, ZEND_ACC_PUBLIC)
  4559. PHP_ME(PharFileInfo, decompress, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4560. PHP_ME(PharFileInfo, delMetadata, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4561. PHP_ME(PharFileInfo, getCompressedSize, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4562. PHP_ME(PharFileInfo, getCRC32, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4563. PHP_ME(PharFileInfo, getContent, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4564. PHP_ME(PharFileInfo, getMetadata, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4565. PHP_ME(PharFileInfo, getPharFlags, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4566. PHP_ME(PharFileInfo, hasMetadata, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4567. PHP_ME(PharFileInfo, isCompressed, arginfo_phar_compo, ZEND_ACC_PUBLIC)
  4568. PHP_ME(PharFileInfo, isCRCChecked, arginfo_phar__void, ZEND_ACC_PUBLIC)
  4569. PHP_ME(PharFileInfo, setMetadata, arginfo_phar_setMetadata, ZEND_ACC_PUBLIC)
  4570. PHP_FE_END
  4571. };
  4572. #endif /* HAVE_SPL */
  4573. zend_function_entry phar_exception_methods[] = {
  4574. PHP_FE_END
  4575. };
  4576. /* }}} */
  4577. #define REGISTER_PHAR_CLASS_CONST_LONG(class_name, const_name, value) \
  4578. zend_declare_class_constant_long(class_name, const_name, sizeof(const_name)-1, (long)value TSRMLS_CC);
  4579. #define phar_exception_get_default() zend_exception_get_default(TSRMLS_C)
  4580. void phar_object_init(TSRMLS_D) /* {{{ */
  4581. {
  4582. zend_class_entry ce;
  4583. INIT_CLASS_ENTRY(ce, "PharException", phar_exception_methods);
  4584. phar_ce_PharException = zend_register_internal_class_ex(&ce, phar_exception_get_default(), NULL TSRMLS_CC);
  4585. #if HAVE_SPL
  4586. INIT_CLASS_ENTRY(ce, "Phar", php_archive_methods);
  4587. phar_ce_archive = zend_register_internal_class_ex(&ce, spl_ce_RecursiveDirectoryIterator, NULL TSRMLS_CC);
  4588. zend_class_implements(phar_ce_archive TSRMLS_CC, 2, spl_ce_Countable, zend_ce_arrayaccess);
  4589. INIT_CLASS_ENTRY(ce, "PharData", php_archive_methods);
  4590. phar_ce_data = zend_register_internal_class_ex(&ce, spl_ce_RecursiveDirectoryIterator, NULL TSRMLS_CC);
  4591. zend_class_implements(phar_ce_data TSRMLS_CC, 2, spl_ce_Countable, zend_ce_arrayaccess);
  4592. INIT_CLASS_ENTRY(ce, "PharFileInfo", php_entry_methods);
  4593. phar_ce_entry = zend_register_internal_class_ex(&ce, spl_ce_SplFileInfo, NULL TSRMLS_CC);
  4594. #else
  4595. INIT_CLASS_ENTRY(ce, "Phar", php_archive_methods);
  4596. phar_ce_archive = zend_register_internal_class(&ce TSRMLS_CC);
  4597. phar_ce_archive->ce_flags |= ZEND_ACC_FINAL_CLASS;
  4598. INIT_CLASS_ENTRY(ce, "PharData", php_archive_methods);
  4599. phar_ce_data = zend_register_internal_class(&ce TSRMLS_CC);
  4600. phar_ce_data->ce_flags |= ZEND_ACC_FINAL_CLASS;
  4601. #endif
  4602. REGISTER_PHAR_CLASS_CONST_LONG(phar_ce_archive, "BZ2", PHAR_ENT_COMPRESSED_BZ2)
  4603. REGISTER_PHAR_CLASS_CONST_LONG(phar_ce_archive, "GZ", PHAR_ENT_COMPRESSED_GZ)
  4604. REGISTER_PHAR_CLASS_CONST_LONG(phar_ce_archive, "NONE", PHAR_ENT_COMPRESSED_NONE)
  4605. REGISTER_PHAR_CLASS_CONST_LONG(phar_ce_archive, "PHAR", PHAR_FORMAT_PHAR)
  4606. REGISTER_PHAR_CLASS_CONST_LONG(phar_ce_archive, "TAR", PHAR_FORMAT_TAR)
  4607. REGISTER_PHAR_CLASS_CONST_LONG(phar_ce_archive, "ZIP", PHAR_FORMAT_ZIP)
  4608. REGISTER_PHAR_CLASS_CONST_LONG(phar_ce_archive, "COMPRESSED", PHAR_ENT_COMPRESSION_MASK)
  4609. REGISTER_PHAR_CLASS_CONST_LONG(phar_ce_archive, "PHP", PHAR_MIME_PHP)
  4610. REGISTER_PHAR_CLASS_CONST_LONG(phar_ce_archive, "PHPS", PHAR_MIME_PHPS)
  4611. REGISTER_PHAR_CLASS_CONST_LONG(phar_ce_archive, "MD5", PHAR_SIG_MD5)
  4612. REGISTER_PHAR_CLASS_CONST_LONG(phar_ce_archive, "OPENSSL", PHAR_SIG_OPENSSL)
  4613. REGISTER_PHAR_CLASS_CONST_LONG(phar_ce_archive, "SHA1", PHAR_SIG_SHA1)
  4614. REGISTER_PHAR_CLASS_CONST_LONG(phar_ce_archive, "SHA256", PHAR_SIG_SHA256)
  4615. REGISTER_PHAR_CLASS_CONST_LONG(phar_ce_archive, "SHA512", PHAR_SIG_SHA512)
  4616. }
  4617. /* }}} */
  4618. /*
  4619. * Local variables:
  4620. * tab-width: 4
  4621. * c-basic-offset: 4
  4622. * End:
  4623. * vim600: noet sw=4 ts=4 fdm=marker
  4624. * vim<600: noet sw=4 ts=4
  4625. */