functions.php 308 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396539753985399540054015402540354045405540654075408540954105411541254135414541554165417541854195420542154225423542454255426542754285429543054315432543354345435543654375438543954405441544254435444544554465447544854495450545154525453545454555456545754585459546054615462546354645465546654675468546954705471547254735474547554765477547854795480548154825483548454855486548754885489549054915492549354945495549654975498549955005501550255035504550555065507550855095510551155125513551455155516551755185519552055215522552355245525552655275528552955305531553255335534553555365537553855395540554155425543554455455546554755485549555055515552555355545555555655575558555955605561556255635564556555665567556855695570557155725573557455755576557755785579558055815582558355845585558655875588558955905591559255935594559555965597559855995600560156025603560456055606560756085609561056115612561356145615561656175618561956205621562256235624562556265627562856295630563156325633563456355636563756385639564056415642564356445645564656475648564956505651565256535654565556565657565856595660566156625663566456655666566756685669567056715672567356745675567656775678567956805681568256835684568556865687568856895690569156925693569456955696569756985699570057015702570357045705570657075708570957105711571257135714571557165717571857195720572157225723572457255726572757285729573057315732573357345735573657375738573957405741574257435744574557465747574857495750575157525753575457555756575757585759576057615762576357645765576657675768576957705771577257735774577557765777577857795780578157825783578457855786578757885789579057915792579357945795579657975798579958005801580258035804580558065807580858095810581158125813581458155816581758185819582058215822582358245825582658275828582958305831583258335834583558365837583858395840584158425843584458455846584758485849585058515852585358545855585658575858585958605861586258635864586558665867586858695870587158725873587458755876587758785879588058815882588358845885588658875888588958905891589258935894589558965897589858995900590159025903590459055906590759085909591059115912591359145915591659175918591959205921592259235924592559265927592859295930593159325933593459355936593759385939594059415942594359445945594659475948594959505951595259535954595559565957595859595960596159625963596459655966596759685969597059715972597359745975597659775978597959805981598259835984598559865987598859895990599159925993599459955996599759985999600060016002600360046005600660076008600960106011601260136014601560166017601860196020602160226023602460256026602760286029603060316032603360346035603660376038603960406041604260436044604560466047604860496050605160526053605460556056605760586059606060616062606360646065606660676068606960706071607260736074607560766077607860796080608160826083608460856086608760886089609060916092609360946095609660976098609961006101610261036104610561066107610861096110611161126113611461156116611761186119612061216122612361246125612661276128612961306131613261336134613561366137613861396140614161426143614461456146614761486149615061516152615361546155615661576158615961606161616261636164616561666167616861696170617161726173617461756176617761786179618061816182618361846185618661876188618961906191619261936194619561966197619861996200620162026203620462056206620762086209621062116212621362146215621662176218621962206221622262236224622562266227622862296230623162326233623462356236623762386239624062416242624362446245624662476248624962506251625262536254625562566257625862596260626162626263626462656266626762686269627062716272627362746275627662776278627962806281628262836284628562866287628862896290629162926293629462956296629762986299630063016302630363046305630663076308630963106311631263136314631563166317631863196320632163226323632463256326632763286329633063316332633363346335633663376338633963406341634263436344634563466347634863496350635163526353635463556356635763586359636063616362636363646365636663676368636963706371637263736374637563766377637863796380638163826383638463856386638763886389639063916392639363946395639663976398639964006401640264036404640564066407640864096410641164126413641464156416641764186419642064216422642364246425642664276428642964306431643264336434643564366437643864396440644164426443644464456446644764486449645064516452645364546455645664576458645964606461646264636464646564666467646864696470647164726473647464756476647764786479648064816482648364846485648664876488648964906491649264936494649564966497649864996500650165026503650465056506650765086509651065116512651365146515651665176518651965206521652265236524652565266527652865296530653165326533653465356536653765386539654065416542654365446545654665476548654965506551655265536554655565566557655865596560656165626563656465656566656765686569657065716572657365746575657665776578657965806581658265836584658565866587658865896590659165926593659465956596659765986599660066016602660366046605660666076608660966106611661266136614661566166617661866196620662166226623662466256626662766286629663066316632663366346635663666376638663966406641664266436644664566466647664866496650665166526653665466556656665766586659666066616662666366646665666666676668666966706671667266736674667566766677667866796680668166826683668466856686668766886689669066916692669366946695669666976698669967006701670267036704670567066707670867096710671167126713671467156716671767186719672067216722672367246725672667276728672967306731673267336734673567366737673867396740674167426743674467456746674767486749675067516752675367546755675667576758675967606761676267636764676567666767676867696770677167726773677467756776677767786779678067816782678367846785678667876788678967906791679267936794679567966797679867996800680168026803680468056806680768086809681068116812681368146815681668176818681968206821682268236824682568266827682868296830683168326833683468356836683768386839684068416842684368446845684668476848684968506851685268536854685568566857685868596860686168626863686468656866686768686869687068716872687368746875687668776878687968806881688268836884688568866887688868896890689168926893689468956896689768986899690069016902690369046905690669076908690969106911691269136914691569166917691869196920692169226923692469256926692769286929693069316932693369346935693669376938693969406941694269436944694569466947694869496950695169526953695469556956695769586959696069616962696369646965696669676968696969706971697269736974697569766977697869796980698169826983698469856986698769886989699069916992699369946995699669976998699970007001700270037004700570067007700870097010701170127013701470157016701770187019702070217022702370247025702670277028702970307031703270337034703570367037703870397040704170427043704470457046704770487049705070517052705370547055705670577058705970607061706270637064706570667067706870697070707170727073707470757076707770787079708070817082708370847085708670877088708970907091709270937094709570967097709870997100710171027103710471057106710771087109711071117112711371147115711671177118711971207121712271237124712571267127712871297130713171327133713471357136713771387139714071417142714371447145714671477148714971507151715271537154715571567157715871597160716171627163716471657166716771687169717071717172717371747175717671777178717971807181718271837184718571867187718871897190719171927193719471957196719771987199720072017202720372047205720672077208720972107211721272137214721572167217721872197220722172227223722472257226722772287229723072317232723372347235723672377238723972407241724272437244724572467247724872497250725172527253725472557256725772587259726072617262726372647265726672677268726972707271727272737274727572767277727872797280728172827283728472857286728772887289729072917292729372947295729672977298729973007301730273037304730573067307730873097310731173127313731473157316731773187319732073217322732373247325732673277328732973307331733273337334733573367337733873397340734173427343734473457346734773487349735073517352735373547355735673577358735973607361736273637364736573667367736873697370737173727373737473757376737773787379738073817382738373847385738673877388738973907391739273937394739573967397739873997400740174027403740474057406740774087409741074117412741374147415741674177418741974207421742274237424742574267427742874297430743174327433743474357436743774387439744074417442744374447445744674477448744974507451745274537454745574567457745874597460746174627463746474657466746774687469747074717472747374747475747674777478747974807481748274837484748574867487
  1. <?php
  2. // ===================================
  3. // Define Version
  4. define('INSTALLEDVERSION', '1.50');
  5. // ===================================
  6. use Kryptonit3\Sonarr\Sonarr;
  7. use Kryptonit3\SickRage\SickRage;
  8. // Debugging output functions
  9. function debug_out($variable, $die = false) {
  10. $trace = debug_backtrace()[0];
  11. echo "<center><img height='200px' src='images/confused.png'></center>";
  12. echo "<center>Look's like something happened, here are the errors and perhaps how to fix them:</center>";
  13. echo '<pre style="white-space: pre-line; background-color: #f2f2f2; border: 2px solid black; border-radius: 5px; padding: 5px; margin: 5px;">'.$trace['file'].':'.$trace['line']."\n\n".print_r($variable, true).'</pre>';
  14. if ($die) { http_response_code(503); die(); }
  15. }
  16. // ==== Auth Plugins START ====
  17. if (function_exists('ldap_connect')) :
  18. // Pass credentials to LDAP backend
  19. function plugin_auth_ldap($username, $password) {
  20. $ldapServers = explode(',',AUTHBACKENDHOST);
  21. foreach($ldapServers as $key => $value) {
  22. // Calculate parts
  23. $digest = parse_url(trim($value));
  24. $scheme = strtolower((isset($digest['scheme'])?$digest['scheme']:'ldap'));
  25. $host = (isset($digest['host'])?$digest['host']:(isset($digest['path'])?$digest['path']:''));
  26. $port = (isset($digest['port'])?$digest['port']:(strtolower($scheme)=='ldap'?389:636));
  27. // Reassign
  28. $ldapServers[$key] = $scheme.'://'.$host.':'.$port;
  29. }
  30. // returns true or false
  31. $ldap = ldap_connect(implode(' ',$ldapServers));
  32. if(empty(AUTHBACKENDDOMAINFORMAT)){
  33. if ($bind = ldap_bind($ldap, AUTHBACKENDDOMAIN.'\\'.$username, $password)) {
  34. writeLog("success", "LDAP authentication success");
  35. return true;
  36. } else {
  37. writeLog("error", "LDAP could not authenticate");
  38. return false;
  39. }
  40. }else{
  41. ldap_set_option($ldap, LDAP_OPT_PROTOCOL_VERSION, 3);
  42. ldap_set_option($ldap, LDAP_OPT_REFERRALS, 0);
  43. $bind = ldap_bind($ldap, sprinf(AUTHBACKENDDOMAINFORMAT, $username), $password);
  44. if ($bind) {
  45. writeLog("success", "LDAP authentication success");
  46. return true;
  47. } else {
  48. writeLog("error", "LDPA could not authenticate");
  49. return false;
  50. }
  51. }
  52. writeLog("error", "LDAP could not authenticate");
  53. return false;
  54. }
  55. else :
  56. // Ldap Auth Missing Dependancy
  57. function plugin_auth_ldap_disabled() {
  58. return 'LDAP - Disabled (Dependancy: php-ldap missing!)';
  59. }
  60. endif;
  61. // Pass credentials to FTP backend
  62. function plugin_auth_ftp($username, $password) {
  63. // Calculate parts
  64. $digest = parse_url(AUTHBACKENDHOST);
  65. $scheme = strtolower((isset($digest['scheme'])?$digest['scheme']:(function_exists('ftp_ssl_connect')?'ftps':'ftp')));
  66. $host = (isset($digest['host'])?$digest['host']:(isset($digest['path'])?$digest['path']:''));
  67. $port = (isset($digest['port'])?$digest['port']:21);
  68. // Determine Connection Type
  69. if ($scheme == 'ftps') {
  70. $conn_id = ftp_ssl_connect($host, $port, 20);
  71. } elseif ($scheme == 'ftp') {
  72. $conn_id = ftp_connect($host, $port, 20);
  73. } else {
  74. debug_out('Invalid FTP scheme. Use ftp or ftps');
  75. writeLog("error", "invalid FTP scheme");
  76. return false;
  77. }
  78. // Check if valid FTP connection
  79. if ($conn_id) {
  80. // Attempt login
  81. @$login_result = ftp_login($conn_id, $username, $password);
  82. ftp_close($conn_id);
  83. // Return Result
  84. if ($login_result) {
  85. writeLog("success", "$username authenticated");
  86. return true;
  87. } else {
  88. writeLog("error", "$username could not authenticate");
  89. return false;
  90. }
  91. } else {
  92. return false;
  93. }
  94. return false;
  95. }
  96. // Pass credentials to Emby Backend
  97. function plugin_auth_emby_local($username, $password) {
  98. $embyAddress = qualifyURL(EMBYURL);
  99. $headers = array(
  100. 'Authorization'=> 'MediaBrowser UserId="e8837bc1-ad67-520e-8cd2-f629e3155721", Client="None", Device="Organizr", DeviceId="xxx", Version="1.0.0.0"',
  101. 'Content-Type' => 'application/json',
  102. );
  103. $body = array(
  104. 'Username' => $username,
  105. 'Password' => sha1($password),
  106. 'PasswordMd5' => md5($password),
  107. );
  108. $response = post_router($embyAddress.'/Users/AuthenticateByName', $body, $headers);
  109. if (isset($response['content'])) {
  110. $json = json_decode($response['content'], true);
  111. if (is_array($json) && isset($json['SessionInfo']) && isset($json['User']) && $json['User']['HasPassword'] == true) {
  112. // Login Success - Now Logout Emby Session As We No Longer Need It
  113. $headers = array(
  114. 'X-Mediabrowser-Token' => $json['AccessToken'],
  115. );
  116. $response = post_router($embyAddress.'/Sessions/Logout', array(), $headers);
  117. return true;
  118. }
  119. }
  120. return false;
  121. }
  122. if (function_exists('curl_version')) :
  123. // Authenticate Against Emby Local (first) and Emby Connect
  124. function plugin_auth_emby_all($username, $password) {
  125. $localResult = plugin_auth_emby_local($username, $password);
  126. if ($localResult) {
  127. return $localResult;
  128. } else {
  129. return plugin_auth_emby_connect($username, $password);
  130. }
  131. }
  132. // Authenicate against emby connect
  133. function plugin_auth_emby_connect($username, $password) {
  134. $embyAddress = qualifyURL(EMBYURL);
  135. // Get A User
  136. $connectId = '';
  137. $userIds = json_decode(file_get_contents($embyAddress.'/Users?api_key='.EMBYTOKEN),true);
  138. if (is_array($userIds)) {
  139. foreach ($userIds as $key => $value) { // Scan for this user
  140. if (isset($value['ConnectUserName']) && isset($value['ConnectUserId'])) { // Qualifty as connect account
  141. if ($value['ConnectUserName'] == $username || $value['Name'] == $username) {
  142. $connectId = $value['ConnectUserId'];
  143. break;
  144. }
  145. }
  146. }
  147. if ($connectId) {
  148. $connectURL = 'https://connect.emby.media/service/user/authenticate';
  149. $headers = array(
  150. 'Accept'=> 'application/json',
  151. 'Content-Type' => 'application/x-www-form-urlencoded',
  152. );
  153. $body = array(
  154. 'nameOrEmail' => $username,
  155. 'rawpw' => $password,
  156. );
  157. $result = curl_post($connectURL, $body, $headers);
  158. if (isset($result['content'])) {
  159. $json = json_decode($result['content'], true);
  160. if (is_array($json) && isset($json['AccessToken']) && isset($json['User']) && $json['User']['Id'] == $connectId) {
  161. return array(
  162. 'email' => $json['User']['Email'],
  163. 'image' => $json['User']['ImageUrl'],
  164. );
  165. }
  166. }
  167. }
  168. }
  169. return false;
  170. }
  171. // Pass credentials to Plex Backend
  172. function plugin_auth_plex($username, $password) {
  173. // Quick out
  174. if ((strtolower(PLEXUSERNAME) == strtolower($username)) && $password == PLEXPASSWORD) {
  175. writeLog("success", $username." authenticated by plex");
  176. return true;
  177. }
  178. //Get User List
  179. $userURL = 'https://plex.tv/pms/friends/all';
  180. $userHeaders = array(
  181. 'Authorization' => 'Basic '.base64_encode(PLEXUSERNAME.':'.PLEXPASSWORD),
  182. );
  183. libxml_use_internal_errors(true);
  184. $userXML = simplexml_load_string(curl_get($userURL, $userHeaders));
  185. if (is_array($userXML) || is_object($userXML)) {
  186. $isUser = false;
  187. $usernameLower = strtolower($username);
  188. foreach($userXML AS $child) {
  189. if(isset($child['username']) && strtolower($child['username']) == $usernameLower) {
  190. $isUser = true;
  191. writeLog("success", $usernameLower." was found in plex friends list");
  192. break;
  193. }
  194. }
  195. if ($isUser) {
  196. //Login User
  197. $connectURL = 'https://plex.tv/users/sign_in.json';
  198. $headers = array(
  199. 'Accept'=> 'application/json',
  200. 'Content-Type' => 'application/x-www-form-urlencoded',
  201. 'X-Plex-Product' => 'Organizr',
  202. 'X-Plex-Version' => '1.0',
  203. 'X-Plex-Client-Identifier' => '01010101-10101010',
  204. );
  205. $body = array(
  206. 'user[login]' => $username,
  207. 'user[password]' => $password,
  208. );
  209. $result = curl_post($connectURL, $body, $headers);
  210. if (isset($result['content'])) {
  211. $json = json_decode($result['content'], true);
  212. if (is_array($json) && isset($json['user']) && isset($json['user']['username']) && strtolower($json['user']['username']) == $usernameLower) {
  213. writeLog("success", $json['user']['username']." was logged into organizr using plex credentials");
  214. return array(
  215. 'email' => $json['user']['email'],
  216. 'image' => $json['user']['thumb']
  217. );
  218. }
  219. }
  220. }else{
  221. writeLog("error", "$username is not an authorized PLEX user or entered invalid password");
  222. }
  223. }else{
  224. writeLog("error", "error occured logging into plex might want to check curl.cainfo=/path/to/downloaded/cacert.pem in php.ini");
  225. }
  226. return false;
  227. }
  228. else :
  229. // Plex Auth Missing Dependancy
  230. function plugin_auth_plex_disabled() {
  231. return 'Plex - Disabled (Dependancy: php-curl missing!)';
  232. }
  233. // Emby Connect Auth Missing Dependancy
  234. function plugin_auth_emby_connect_disabled() {
  235. return 'Emby Connect - Disabled (Dependancy: php-curl missing!)';
  236. }
  237. // Emby Both Auth Missing Dependancy
  238. function plugin_auth_emby_both_disabled() {
  239. return 'Emby Both - Disabled (Dependancy: php-curl missing!)';
  240. }
  241. endif;
  242. // ==== Auth Plugins END ====
  243. // ==== General Class Definitions START ====
  244. $userLanguage = isset($_SERVER['HTTP_ACCEPT_LANGUAGE']) ? substr($_SERVER['HTTP_ACCEPT_LANGUAGE'], 0, 2) : "en";
  245. class setLanguage {
  246. private $language = null;
  247. private $langCode = null;
  248. function __construct($language = false) {
  249. // Default
  250. if (!$language) {
  251. $language = isset($_SERVER['HTTP_ACCEPT_LANGUAGE']) ? substr($_SERVER['HTTP_ACCEPT_LANGUAGE'], 0, 2) : "en";
  252. }
  253. if (!file_exists("lang/{$language}.ini")) {
  254. $language = 'en';
  255. }
  256. $this->langCode = $language;
  257. $this->language = parse_ini_file("lang/{$language}.ini", false, INI_SCANNER_RAW);
  258. if (file_exists("lang/{$language}.cust.ini")) {
  259. foreach($tmp = parse_ini_file("lang/{$language}.cust.ini", false, INI_SCANNER_RAW) as $k => $v) {
  260. $this->language[$k] = $v;
  261. }
  262. }
  263. }
  264. public function getLang() {
  265. return $this->langCode;
  266. }
  267. public function translate($originalWord) {
  268. $getArg = func_num_args();
  269. if ($getArg > 1) {
  270. $allWords = func_get_args();
  271. array_shift($allWords);
  272. } else {
  273. $allWords = array();
  274. }
  275. $translatedWord = isset($this->language[$originalWord]) ? $this->language[$originalWord] : null;
  276. if (!$translatedWord) {
  277. return ucwords(str_replace("_", " ", strtolower($originalWord)));
  278. //echo "WHOA!!!!!!! $originalWord";
  279. }
  280. $translatedWord = htmlspecialchars($translatedWord, ENT_QUOTES);
  281. return vsprintf($translatedWord, $allWords);
  282. }
  283. }
  284. $language = new setLanguage;
  285. // ==== General Class Definitions END ====
  286. // Direct request to curl if it exists, otherwise handle if not HTTPS
  287. function post_router($url, $data, $headers = array(), $referer='') {
  288. if (function_exists('curl_version')) {
  289. return curl_post($url, $data, $headers, $referer);
  290. } else {
  291. return post_request($url, $data, $headers, $referer);
  292. }
  293. }
  294. if (function_exists('curl_version')) :
  295. // Curl Post
  296. function curl_post($url, $data, $headers = array(), $referer='') {
  297. // Initiate cURL
  298. $curlReq = curl_init($url);
  299. // As post request
  300. curl_setopt($curlReq, CURLOPT_CUSTOMREQUEST, "POST");
  301. curl_setopt($curlReq, CURLOPT_RETURNTRANSFER, true);
  302. curl_setopt($curlReq, CURLOPT_CAINFO, getCert());
  303. if(localURL($url)){
  304. curl_setopt($curlReq, CURLOPT_SSL_VERIFYHOST, 0);
  305. curl_setopt($curlReq, CURLOPT_SSL_VERIFYPEER, 0);
  306. }
  307. // Format Data
  308. switch (isset($headers['Content-Type'])?$headers['Content-Type']:'') {
  309. case 'application/json':
  310. curl_setopt($curlReq, CURLOPT_POSTFIELDS, json_encode($data));
  311. break;
  312. case 'application/x-www-form-urlencoded';
  313. curl_setopt($curlReq, CURLOPT_POSTFIELDS, http_build_query($data));
  314. break;
  315. default:
  316. $headers['Content-Type'] = 'application/x-www-form-urlencoded';
  317. curl_setopt($curlReq, CURLOPT_POSTFIELDS, http_build_query($data));
  318. }
  319. // Format Headers
  320. $cHeaders = array();
  321. foreach ($headers as $k => $v) {
  322. $cHeaders[] = $k.': '.$v;
  323. }
  324. if (count($cHeaders)) {
  325. curl_setopt($curlReq, CURLOPT_HTTPHEADER, $cHeaders);
  326. }
  327. // Execute
  328. $result = curl_exec($curlReq);
  329. $httpcode = curl_getinfo($curlReq);
  330. // Close
  331. curl_close($curlReq);
  332. // Return
  333. return array('content'=>$result, 'http_code'=>$httpcode);
  334. }
  335. //Curl Get Function
  336. function curl_get($url, $headers = array()) {
  337. // Initiate cURL
  338. $curlReq = curl_init($url);
  339. // As post request
  340. curl_setopt($curlReq, CURLOPT_CUSTOMREQUEST, "GET");
  341. curl_setopt($curlReq, CURLOPT_RETURNTRANSFER, true);
  342. curl_setopt($curlReq, CURLOPT_CAINFO, getCert());
  343. curl_setopt($curlReq, CURLOPT_CONNECTTIMEOUT, 5);
  344. if(localURL($url)){
  345. curl_setopt($curlReq, CURLOPT_SSL_VERIFYHOST, 0);
  346. curl_setopt($curlReq, CURLOPT_SSL_VERIFYPEER, 0);
  347. }
  348. // Format Headers
  349. $cHeaders = array();
  350. foreach ($headers as $k => $v) {
  351. $cHeaders[] = $k.': '.$v;
  352. }
  353. if (count($cHeaders)) {
  354. curl_setopt($curlReq, CURLOPT_HTTPHEADER, $cHeaders);
  355. }
  356. // Execute
  357. $result = curl_exec($curlReq);
  358. // Close
  359. curl_close($curlReq);
  360. // Return
  361. return $result;
  362. }
  363. //Curl Delete Function
  364. function curl_delete($url, $headers = array()) {
  365. // Initiate cURL
  366. $curlReq = curl_init($url);
  367. // As post request
  368. curl_setopt($curlReq, CURLOPT_CUSTOMREQUEST, "DELETE");
  369. curl_setopt($curlReq, CURLOPT_RETURNTRANSFER, true);
  370. curl_setopt($curlReq, CURLOPT_CONNECTTIMEOUT, 5);
  371. curl_setopt($curlReq, CURLOPT_CAINFO, getCert());
  372. if(localURL($url)){
  373. curl_setopt($curlReq, CURLOPT_SSL_VERIFYHOST, 0);
  374. curl_setopt($curlReq, CURLOPT_SSL_VERIFYPEER, 0);
  375. }
  376. // Format Headers
  377. $cHeaders = array();
  378. foreach ($headers as $k => $v) {
  379. $cHeaders[] = $k.': '.$v;
  380. }
  381. if (count($cHeaders)) {
  382. curl_setopt($curlReq, CURLOPT_HTTPHEADER, $cHeaders);
  383. }
  384. // Execute
  385. $result = curl_exec($curlReq);
  386. $httpcode = curl_getinfo($curlReq);
  387. // Close
  388. curl_close($curlReq);
  389. // Return
  390. return array('content'=>$result, 'http_code'=>$httpcode);
  391. }
  392. endif;
  393. //Case-Insensitive Function
  394. function in_arrayi($needle, $haystack) {
  395. return in_array(strtolower($needle), array_map('strtolower', $haystack));
  396. }
  397. // HTTP post request (Removes need for curl, probably useless)
  398. function post_request($url, $data, $headers = array(), $referer='') {
  399. // Adapted from http://stackoverflow.com/a/28387011/6810513
  400. // Convert the data array into URL Parameters like a=b&foo=bar etc.
  401. if (isset($headers['Content-Type'])) {
  402. switch ($headers['Content-Type']) {
  403. case 'application/json':
  404. $data = json_encode($data);
  405. break;
  406. case 'application/x-www-form-urlencoded':
  407. $data = http_build_query($data);
  408. break;
  409. }
  410. } else {
  411. $headers['Content-Type'] = 'application/x-www-form-urlencoded';
  412. $data = http_build_query($data);
  413. }
  414. // parse the given URL
  415. $urlDigest = parse_url($url);
  416. // extract host and path:
  417. $host = $urlDigest['host'];
  418. $path = $urlDigest['path'];
  419. if ($urlDigest['scheme'] != 'http') {
  420. die('Error: Only HTTP request are supported, please use cURL to add HTTPS support! ('.$urlDigest['scheme'].'://'.$host.')');
  421. }
  422. // open a socket connection on port 80 - timeout: 30 sec
  423. $fp = fsockopen($host, (isset($urlDigest['port'])?':'.$urlDigest['port']:80), $errno, $errstr, 30);
  424. if ($fp){
  425. // send the request headers:
  426. fputs($fp, "POST $path HTTP/1.1\r\n");
  427. fputs($fp, "Host: $host\r\n");
  428. if ($referer != '')
  429. fputs($fp, "Referer: $referer\r\n");
  430. fputs($fp, "Content-length: ". strlen($data) ."\r\n");
  431. foreach($headers as $k => $v) {
  432. fputs($fp, $k.": ".$v."\r\n");
  433. }
  434. fputs($fp, "Connection: close\r\n\r\n");
  435. fputs($fp, $data);
  436. $result = '';
  437. while(!feof($fp)) {
  438. // receive the results of the request
  439. $result .= fgets($fp, 128);
  440. }
  441. }
  442. else {
  443. return array(
  444. 'status' => 'err',
  445. 'error' => "$errstr ($errno)"
  446. );
  447. }
  448. // close the socket connection:
  449. fclose($fp);
  450. // split the result header from the content
  451. $result = explode("\r\n\r\n", $result, 2);
  452. $header = isset($result[0]) ? $result[0] : '';
  453. $content = isset($result[1]) ? $result[1] : '';
  454. // return as structured array:
  455. return array(
  456. 'status' => 'ok',
  457. 'header' => $header,
  458. 'content' => $content,
  459. );
  460. }
  461. // Format item from Emby for Carousel
  462. function resolveEmbyItem($address, $token, $item, $nowPlaying = false, $showNames = false, $role = false, $moreInfo = false) {
  463. // Static Height
  464. $height = 444;
  465. // Get Item Details
  466. $itemDetails = json_decode(file_get_contents($address.'/Items?Ids='.$item['Id'].'&api_key='.$token),true)['Items'][0];
  467. /*if (substr_count(EMBYURL, ':') == 2) {
  468. $URL = "http://app.emby.media/itemdetails.html?id=".$itemDetails['Id'];
  469. }else{
  470. $URL = EMBYURL."/web/itemdetails.html?id=".$itemDetails['Id'];
  471. }*/
  472. $URL = EMBYURL."/web/itemdetails.html?id=".$itemDetails['Id'];
  473. switch ($itemDetails['Type']) {
  474. case 'Episode':
  475. $title = (isset($itemDetails['SeriesName'])?$itemDetails['SeriesName']:"");
  476. $imageId = (isset($itemDetails['SeriesId'])?$itemDetails['SeriesId']:$itemDetails['Id']);
  477. $width = 300;
  478. $style = '';
  479. $image = 'slick-image-tall';
  480. if(!$nowPlaying){
  481. $imageType = (isset($itemDetails['ImageTags']['Primary']) ? "Primary" : false);
  482. $key = $itemDetails['Id'] . "-list";
  483. }else{
  484. $height = 281;
  485. $width = 500;
  486. $imageId = isset($itemDetails['ParentThumbItemId']) ? $itemDetails['ParentThumbItemId'] : (isset($itemDetails['ParentBackdropItemId']) ? $itemDetails['ParentBackdropItemId'] : false);
  487. $imageType = isset($itemDetails['ParentThumbItemId']) ? "Thumb" : (isset($itemDetails['ParentBackdropItemId']) ? "Backdrop" : false);
  488. $key = (isset($itemDetails['ParentThumbItemId']) ? $itemDetails['ParentThumbItemId']."-np" : "none-np");
  489. $elapsed = $moreInfo['PlayState']['PositionTicks'];
  490. $duration = $moreInfo['NowPlayingItem']['RunTimeTicks'];
  491. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  492. //$transcoded = floor($item->TranscodeSession['progress']- $watched);
  493. $stream = $moreInfo['PlayState']['PlayMethod'];
  494. $user = $role == "admin" ? $moreInfo['UserName'] : "";
  495. $id = $moreInfo['DeviceId'];
  496. $streamInfo = buildStream(array(
  497. 'platform' => (string) $moreInfo['Client'],
  498. 'device' => (string) $moreInfo['DeviceName'],
  499. 'stream' => streamType($stream),
  500. 'video' => streamType($stream)." ".embyArray($moreInfo['NowPlayingItem']['MediaStreams'], "video"),
  501. 'audio' => streamType($stream)." ".embyArray($moreInfo['NowPlayingItem']['MediaStreams'], "audio"),
  502. ));
  503. $state = (($moreInfo['PlayState']['IsPaused'] == "1") ? "pause" : "play");
  504. $topTitle = '<h5 class="text-center zero-m elip">'.$title.' - '.$itemDetails['Name'].'</h5>';
  505. $bottomTitle = '<small class="zero-m">S'.$itemDetails['ParentIndexNumber'].' · E'.$itemDetails['IndexNumber'].'</small>';
  506. if($showNames == "true"){ $bottomTitle .= '</small><small class="zero-m pull-right">'.$user.'</small>'; }
  507. }
  508. break;
  509. case 'MusicAlbum':
  510. case 'Audio':
  511. $title = $itemDetails['Name'];
  512. $imageId = $itemDetails['Id'];
  513. $width = 444;
  514. $style = '';
  515. $image = 'slick-image-short';
  516. if(!$nowPlaying){
  517. $imageType = (isset($itemDetails['ImageTags']['Primary']) ? "Primary" : false);
  518. $key = $itemDetails['Id'] . "-list";
  519. }else{
  520. $height = 281;
  521. $width = 500;
  522. $imageId = (isset($itemDetails['ParentBackdropItemId']) ? $itemDetails['ParentBackdropItemId'] : false);
  523. $imageType = (isset($itemDetails['ParentBackdropItemId']) ? "Backdrop" : false);
  524. $key = (isset($itemDetails['ParentBackdropItemId']) ? $itemDetails['ParentBackdropItemId'] : "no-np") . "-np";
  525. $elapsed = $moreInfo['PlayState']['PositionTicks'];
  526. $duration = $moreInfo['NowPlayingItem']['RunTimeTicks'];
  527. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  528. //$transcoded = floor($item->TranscodeSession['progress']- $watched);
  529. $stream = $moreInfo['PlayState']['PlayMethod'];
  530. $user = $role == "admin" ? $moreInfo['UserName'] : "";
  531. $id = $moreInfo['DeviceId'];
  532. $streamInfo = buildStream(array(
  533. 'platform' => (string) $moreInfo['Client'],
  534. 'device' => (string) $moreInfo['DeviceName'],
  535. 'stream' => streamType($stream),
  536. 'audio' => streamType($stream)." ".embyArray($moreInfo['NowPlayingItem']['MediaStreams'], "audio"),
  537. ));
  538. $state = (($moreInfo['PlayState']['IsPaused'] == "1") ? "pause" : "play");
  539. $topTitle = '<h5 class="text-center zero-m elip">'.$itemDetails['AlbumArtist'].' - '.$itemDetails['Album'].'</h5>';
  540. $bottomTitle = '<small class="zero-m">'.$title.'</small>';
  541. if($showNames == "true"){ $bottomTitle .= '</small><small class="zero-m pull-right">'.$user.'</small>'; }
  542. }
  543. break;
  544. case 'TvChannel':
  545. $title = $itemDetails['CurrentProgram']['Name'];
  546. $imageId = $itemDetails['Id'];
  547. $width = 300;
  548. $style = '';
  549. $image = 'slick-image-tall';
  550. if(!$nowPlaying){
  551. $imageType = "Primary";
  552. $key = $itemDetails['Id'] . "-list";
  553. }else{
  554. $height = 281;
  555. $width = 500;
  556. $imageType = "Thumb";
  557. $key = $itemDetails['Id'] . "-np";
  558. $useImage = "images/livetv.png";
  559. $watched = "0";
  560. $stream = $moreInfo['PlayState']['PlayMethod'];
  561. $user = $role == "admin" ? $moreInfo['UserName'] : "";
  562. $id = $moreInfo['DeviceId'];
  563. $streamInfo = buildStream(array(
  564. 'platform' => (string) $moreInfo['Client'],
  565. 'device' => (string) $moreInfo['DeviceName'],
  566. 'stream' => streamType($stream),
  567. 'video' => streamType($stream)." ".embyArray($moreInfo['NowPlayingItem']['MediaStreams'], "video"),
  568. 'audio' => streamType($stream)." ".embyArray($moreInfo['NowPlayingItem']['MediaStreams'], "audio"),
  569. ));
  570. $state = (($moreInfo['PlayState']['IsPaused'] == "1") ? "pause" : "play");
  571. $topTitle = '<h5 class="text-center zero-m elip">'.$title.'</h5>';
  572. $bottomTitle = '<small class="zero-m">'.$itemDetails['Name'].' - '.$itemDetails['ChannelNumber'].'</small>';
  573. if($showNames == "true"){ $bottomTitle .= '</small><small class="zero-m pull-right">'.$user.'</small>'; }
  574. }
  575. break;
  576. default:
  577. $title = $itemDetails['Name'];
  578. $imageId = $itemDetails['Id'];
  579. $width = 300;
  580. $style = '';
  581. $image = 'slick-image-tall';
  582. if(!$nowPlaying){
  583. $imageType = (isset($itemDetails['ImageTags']['Primary']) ? "Primary" : false);
  584. $key = $itemDetails['Id'] . "-list";
  585. }else{
  586. $height = 281;
  587. $width = 500;
  588. $imageType = isset($itemDetails['ImageTags']['Thumb']) ? "Thumb" : (isset($itemDetails['BackdropImageTags']) ? "Backdrop" : false);
  589. $key = $itemDetails['Id'] . "-np";
  590. $elapsed = $moreInfo['PlayState']['PositionTicks'];
  591. $duration = $moreInfo['NowPlayingItem']['RunTimeTicks'];
  592. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  593. //$transcoded = floor($item->TranscodeSession['progress']- $watched);
  594. $stream = $moreInfo['PlayState']['PlayMethod'];
  595. $user = $role == "admin" ? $moreInfo['UserName'] : "";
  596. $id = $moreInfo['DeviceId'];
  597. $streamInfo = buildStream(array(
  598. 'platform' => (string) $moreInfo['Client'],
  599. 'device' => (string) $moreInfo['DeviceName'],
  600. 'stream' => streamType($stream),
  601. 'video' => streamType($stream)." ".embyArray($moreInfo['NowPlayingItem']['MediaStreams'], "video"),
  602. 'audio' => streamType($stream)." ".embyArray($moreInfo['NowPlayingItem']['MediaStreams'], "audio"),
  603. ));
  604. $state = (($moreInfo['PlayState']['IsPaused'] == "1") ? "pause" : "play");
  605. $topTitle = '<h5 class="text-center zero-m elip">'.$title.'</h5>';
  606. $bottomTitle = '<small class="zero-m">'.$moreInfo['NowPlayingItem']['ProductionYear'].'</small>';
  607. if($showNames == "true"){ $bottomTitle .= '</small><small class="zero-m pull-right">'.$user.'</small>'; }
  608. }
  609. }
  610. // If No Overview
  611. if (!isset($itemDetails['Overview'])) {
  612. $itemDetails['Overview'] = '';
  613. }
  614. if (file_exists('images/cache/'.$key.'.jpg')){ $image_url = 'images/cache/'.$key.'.jpg'; }
  615. if (file_exists('images/cache/'.$key.'.jpg') && (time() - 604800) > filemtime('images/cache/'.$key.'.jpg') || !file_exists('images/cache/'.$key.'.jpg')) {
  616. $image_url = 'ajax.php?a=emby-image&type='.$imageType.'&img='.$imageId.'&height='.$height.'&width='.$width.'&key='.$key.'';
  617. }
  618. if($nowPlaying){
  619. if(!$imageType){ $image_url = "images/no-np.png"; $key = "no-np"; }
  620. if(!$imageId){ $image_url = "images/no-np.png"; $key = "no-np"; }
  621. }else{
  622. if(!$imageType){ $image_url = "images/no-list.png"; $key = "no-list"; }
  623. if(!$imageId){ $image_url = "images/no-list.png"; $key = "no-list"; }
  624. }
  625. if(isset($useImage)){ $image_url = $useImage; }
  626. // Assemble Item And Cache Into Array
  627. if($nowPlaying){
  628. //prettyPrint($itemDetails);
  629. return '<div class="col-sm-6 col-md-3"><div class="thumbnail ultra-widget"><div style="display: none;" np="'.$id.'" class="overlay content-box small-box gray-bg">'.$streamInfo.'</div><span class="w-refresh w-p-icon gray" link="'.$id.'"><span class="fa-stack fa-lg" style="font-size: .5em"><i class="fa fa-square fa-stack-2x"></i><i class="fa fa-info-circle fa-stack-1x fa-inverse"></i></span></span><a href="'.$URL.'" target="_blank"><img style="width: 100%; display:inherit;" src="'.$image_url.'" alt="'.$itemDetails['Name'].'"></a><div class="progress progress-bar-sm zero-m"><div class="progress-bar progress-bar-success" role="progressbar" aria-valuenow="'.$watched.'" aria-valuemin="0" aria-valuemax="100" style="width: '.$watched.'%"></div><div class="progress-bar palette-Grey-500 bg" style="width: 0%"></div></div><div class="caption"><i style="float:left" class="fa fa-'.$state.'"></i>'.$topTitle.''.$bottomTitle.'</div></div></div>';
  630. }else{
  631. return '<div class="item-'.$itemDetails['Type'].'"><a href="'.$URL.'" target="_blank"><img alt="'.$itemDetails['Name'].'" class="'.$image.'" data-lazy="'.$image_url.'"></a><small style="margin-right: 13px" class="elip">'.$title.'</small></div>';
  632. }
  633. }
  634. // Format item from Plex for Carousel
  635. function resolvePlexItem($server, $token, $item, $nowPlaying = false, $showNames = false, $role = false, $playlist = false) {
  636. // Static Height
  637. $height = 444;
  638. $playlist = ($playlist) ? " playlist-$playlist" : "";
  639. switch ($item['type']) {
  640. case 'season':
  641. $title = $item['parentTitle'];
  642. $summary = $item['parentSummary'];
  643. $width = 300;
  644. $image = 'slick-image-tall';
  645. $style = '';
  646. if(!$nowPlaying){
  647. $thumb = $item['thumb'];
  648. $key = $item['ratingKey'] . "-list";
  649. }else {
  650. $height = 281;
  651. $width = 500;
  652. $thumb = $item['art'];
  653. $key = $item['ratingKey'] . "-np";
  654. $elapsed = $item['viewOffset'];
  655. $duration = ($item['duration']) ? $item['duration'] : $item->Media['duration'];
  656. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  657. $transcoded = floor($item->TranscodeSession['progress']- $watched);
  658. $stream = $item->Media->Part->Stream['decision'];
  659. $user = $role == "admin" ? $item->User['title'] : "";
  660. $id = str_replace('"', '', $item->Player['machineIdentifier']);
  661. $streamInfo = buildStream(array(
  662. 'platform' => (string) $item->Player['platform'],
  663. 'device' => (string) $item->Player['device'],
  664. 'stream' => streamType($item->Media->Part['decision']),
  665. 'video' => streamType($item->Media->Part->Stream[0]['decision'])." (".$item->Media->Part->Stream[0]['codec'].") (".$item->Media->Part->Stream[0]['width']."x".$item->Media->Part->Stream[0]['height'].")",
  666. 'audio' => streamType($item->Media->Part->Stream[1]['decision'])." (".$item->Media->Part->Stream[1]['codec'].") (".$item->Media->Part->Stream[1]['channels']."ch)",
  667. ));
  668. $state = (($item->Player['state'] == "paused") ? "pause" : "play");
  669. }
  670. break;
  671. case 'episode':
  672. $title = $item['grandparentTitle'];
  673. $summary = $item['title'];
  674. $width = 300;
  675. $image = 'slick-image-tall';
  676. $style = '';
  677. if(!$nowPlaying){
  678. $thumb = ($item['parentThumb'] ? $item['parentThumb'] : $item['grandparentThumb']);
  679. $key = $item['ratingKey'] . "-list";
  680. }else {
  681. $height = 281;
  682. $width = 500;
  683. $thumb = $item['art'];
  684. $key = $item['ratingKey'] . "-np";
  685. $elapsed = $item['viewOffset'];
  686. $duration = ($item['duration']) ? $item['duration'] : $item->Media['duration'];
  687. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  688. $transcoded = floor($item->TranscodeSession['progress']- $watched);
  689. $stream = $item->Media->Part->Stream['decision'];
  690. $user = $role == "admin" ? $item->User['title'] : "";
  691. $id = str_replace('"', '', $item->Player['machineIdentifier']);
  692. $streamInfo = buildStream(array(
  693. 'platform' => (string) $item->Player['platform'],
  694. 'device' => (string) $item->Player['device'],
  695. 'stream' => streamType($item->Media->Part['decision']),
  696. 'video' => streamType($item->Media->Part->Stream[0]['decision'])." (".$item->Media->Part->Stream[0]['codec'].") (".$item->Media->Part->Stream[0]['width']."x".$item->Media->Part->Stream[0]['height'].")",
  697. 'audio' => streamType($item->Media->Part->Stream[1]['decision'])." (".$item->Media->Part->Stream[1]['codec'].") (".$item->Media->Part->Stream[1]['channels']."ch)",
  698. ));
  699. $state = (($item->Player['state'] == "paused") ? "pause" : "play");
  700. $topTitle = '<h5 class="text-center zero-m elip">'.$title.' - '.$item['title'].'</h5>';
  701. $bottomTitle = '<small class="zero-m">S'.$item['parentIndex'].' · E'.$item['index'].'</small>';
  702. if($showNames == "true"){ $bottomTitle .= '<small class="zero-m pull-right">'.$user.'</small>'; }
  703. }
  704. break;
  705. case 'clip':
  706. $title = $item['title'];
  707. $summary = $item['summary'];
  708. $width = 300;
  709. $image = 'slick-image-tall';
  710. $style = '';
  711. if(!$nowPlaying){
  712. $thumb = $item['thumb'];
  713. $key = $item['ratingKey'] . "-list";
  714. }else {
  715. $height = 281;
  716. $width = 500;
  717. $thumb = $item['art'];
  718. $key = isset($item['ratingKey']) ? $item['ratingKey'] . "-np" : (isset($item['live']) ? "livetv.png" : ":)");
  719. $useImage = (isset($item['live']) ? "images/livetv.png" : null);
  720. $extraInfo = isset($item['extraType']) ? "Trailer" : (isset($item['live']) ? "Live TV" : ":)");
  721. $elapsed = $item['viewOffset'];
  722. $duration = ($item['duration']) ? $item['duration'] : $item->Media['duration'];
  723. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  724. $transcoded = floor($item->TranscodeSession['progress']- $watched);
  725. $stream = $item->Media->Part->Stream['decision'];
  726. $user = $role == "admin" ? $item->User['title'] : "";
  727. $id = str_replace('"', '', $item->Player['machineIdentifier']);
  728. $streamInfo = buildStream(array(
  729. 'platform' => (string) $item->Player['platform'],
  730. 'device' => (string) $item->Player['device'],
  731. 'stream' => streamType($item->Media->Part['decision']),
  732. 'video' => streamType($item->Media->Part->Stream[0]['decision'])." (".$item->Media->Part->Stream[0]['codec'].") (".$item->Media->Part->Stream[0]['width']."x".$item->Media->Part->Stream[0]['height'].")",
  733. 'audio' => streamType($item->Media->Part->Stream[1]['decision'])." (".$item->Media->Part->Stream[1]['codec'].") (".$item->Media->Part->Stream[1]['channels']."ch)",
  734. ));
  735. $state = (($item->Player['state'] == "paused") ? "pause" : "play");
  736. $topTitle = '<h5 class="text-center zero-m elip">'.$title.'</h5>';
  737. $bottomTitle = '<small class="zero-m">'.$extraInfo.'</small>';
  738. if($showNames == "true"){ $bottomTitle .= '<small class="zero-m pull-right">'.$user.'</small>'; }
  739. }
  740. break;
  741. case 'album':
  742. case 'track':
  743. $title = $item['parentTitle'];
  744. $summary = $item['title'];
  745. $image = 'slick-image-short';
  746. $style = 'left: 160px !important;';
  747. $item['ratingKey'] = $item['parentRatingKey'];
  748. if(!$nowPlaying){
  749. $width = 444;
  750. $thumb = $item['thumb'];
  751. $key = $item['ratingKey'] . "-list";
  752. }else {
  753. $height = 281;
  754. $width = 500;
  755. $thumb = $item['art'];
  756. $key = $item['ratingKey'] . "-np";
  757. $elapsed = $item['viewOffset'];
  758. $duration = ($item['duration']) ? $item['duration'] : $item->Media['duration'];
  759. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  760. $transcoded = floor($item->TranscodeSession['progress']- $watched);
  761. $stream = $item->Media->Part->Stream['decision'];
  762. $user = $role == "admin" ? $item->User['title'] : "";
  763. $id = str_replace('"', '', $item->Player['machineIdentifier']);
  764. $streamInfo = buildStream(array(
  765. 'platform' => (string) $item->Player['platform'],
  766. 'device' => (string) $item->Player['device'],
  767. 'stream' => streamType($item->Media->Part['decision']),
  768. 'audio' => streamType($item->Media->Part->Stream[0]['decision'])." (".$item->Media->Part->Stream[0]['codec'].") (".$item->Media->Part->Stream[0]['channels']."ch)",
  769. ));
  770. $state = (($item->Player['state'] == "paused") ? "pause" : "play");
  771. $topTitle = '<h5 class="text-center zero-m elip">'.$item['grandparentTitle'].' - '.$item['title'].'</h5>';
  772. $bottomTitle = '<small class="zero-m">'.$title.'</small>';
  773. if($showNames == "true"){ $bottomTitle .= '<small class="zero-m pull-right">'.$user.'</small>'; }
  774. }
  775. break;
  776. default:
  777. $title = $item['title'];
  778. $summary = $item['summary'];
  779. $image = 'slick-image-tall';
  780. $style = '';
  781. if(!$nowPlaying){
  782. $width = 300;
  783. $thumb = $item['thumb'];
  784. $key = $item['ratingKey'] . "-list";
  785. }else {
  786. $height = 281;
  787. $width = 500;
  788. $thumb = $item['art'];
  789. $key = $item['ratingKey'] . "-np";
  790. $elapsed = $item['viewOffset'];
  791. $duration = ($item['duration']) ? $item['duration'] : $item->Media['duration'];
  792. $watched = (!empty($elapsed) ? floor(($elapsed / $duration) * 100) : 0);
  793. $transcoded = floor($item->TranscodeSession['progress']- $watched);
  794. $stream = $item->Media->Part->Stream['decision'];
  795. $user = $role == "admin" ? $item->User['title'] : "";
  796. $id = str_replace('"', '', $item->Player['machineIdentifier']);
  797. $streamInfo = buildStream(array(
  798. 'platform' => (string) $item->Player['platform'],
  799. 'device' => (string) $item->Player['device'],
  800. 'stream' => streamType($item->Media->Part['decision']),
  801. 'video' => streamType($item->Media->Part->Stream[0]['decision'])." (".$item->Media->Part->Stream[0]['codec'].") (".$item->Media->Part->Stream[0]['width']."x".$item->Media->Part->Stream[0]['height'].")",
  802. 'audio' => streamType($item->Media->Part->Stream[1]['decision'])." (".$item->Media->Part->Stream[1]['codec'].") (".$item->Media->Part->Stream[1]['channels']."ch)",
  803. ));
  804. $state = (($item->Player['state'] == "paused") ? "pause" : "play");
  805. $topTitle = '<h5 class="text-center zero-m elip">'.$title.'</h5>';
  806. $bottomTitle = '<small class="zero-m">'.$item['year'].'</small>';
  807. if($showNames == "true"){ $bottomTitle .= '<small class="zero-m pull-right">'.$user.'</small>'; }
  808. }
  809. }
  810. if (substr_count(PLEXURL, '.') != 2) {
  811. $address = "https://app.plex.tv/web/app#!/server/$server/details?key=/library/metadata/".$item['ratingKey'];
  812. }else{
  813. $address = PLEXURL."/web/index.html#!/server/$server/details?key=/library/metadata/".$item['ratingKey'];
  814. }
  815. // If No Overview
  816. if (!isset($itemDetails['Overview'])) { $itemDetails['Overview'] = ''; }
  817. if (file_exists('images/cache/'.$key.'.jpg')){ $image_url = 'images/cache/'.$key.'.jpg'; }
  818. if (file_exists('images/cache/'.$key.'.jpg') && (time() - 604800) > filemtime('images/cache/'.$key.'.jpg') || !file_exists('images/cache/'.$key.'.jpg')) {
  819. $image_url = 'ajax.php?a=plex-image&img='.$thumb.'&height='.$height.'&width='.$width.'&key='.$key.'';
  820. }
  821. if($nowPlaying){
  822. if(!$thumb){ $image_url = "images/no-np.png"; $key = "no-np"; }
  823. }else{
  824. if(!$thumb){ $image_url = "images/no-list.png"; $key = "no-list"; }
  825. }
  826. if(isset($useImage)){ $image_url = $useImage; }
  827. $openTab = (PLEXTABNAME) ? "true" : "false";
  828. // Assemble Item And Cache Into Array
  829. if($nowPlaying){
  830. return '<div class="col-sm-6 col-md-3"><div class="thumbnail ultra-widget"><div style="display: none;" np="'.$id.'" class="overlay content-box small-box gray-bg">'.$streamInfo.'</div><span class="w-refresh w-p-icon gray" link="'.$id.'"><span class="fa-stack fa-lg" style="font-size: .5em"><i class="fa fa-square fa-stack-2x"></i><i class="fa fa-info-circle fa-stack-1x fa-inverse"></i></span></span><a class="openTab" extraTitle="'.$title.'" extraType="'.$item['type'].'" openTab="'.$openTab.'" href="'.$address.'" target="_blank"><img style="width: 100%; display:inherit;" src="'.$image_url.'" alt="'.$item['Name'].'"></a><div class="progress progress-bar-sm zero-m"><div class="progress-bar progress-bar-success" role="progressbar" aria-valuenow="'.$watched.'" aria-valuemin="0" aria-valuemax="100" style="width: '.$watched.'%"></div><div class="progress-bar palette-Grey-500 bg" style="width: '.$transcoded.'%"></div></div><div class="caption"><i style="float:left" class="fa fa-'.$state.'"></i>'.$topTitle.''.$bottomTitle.'</div></div></div>';
  831. }else{
  832. return '<div class="item-'.$item['type'].$playlist.'"><a class="openTab" extraTitle="'.$title.'" extraType="'.$item['type'].'" openTab="'.$openTab.'" href="'.$address.'" target="_blank"><img alt="'.$item['Name'].'" class="'.$image.'" data-lazy="'.$image_url.'"></a><small style="margin-right: 13px" class="elip">'.$title.'</small></div>';
  833. }
  834. }
  835. //$hideMenu .= '<li data-filter="playlist-'.$className.'" data-name="'.$api['title'].'"><a class="js-filter-'.$className.'" href="javascript:void(0)">'.$api['title'].'</a></li>';
  836. //Recent Added
  837. function outputRecentAdded($header, $items, $script = false, $array) {
  838. $hideMenu = '<div class="pull-right"><div class="btn-group" role="group"><button type="button" class="btn waves btn-default btn-sm dropdown-toggle waves-effect waves-float" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">Filter &nbsp;<span class="caret"></span></button><ul style="right:0; left: auto" class="dropdown-menu filter-recent-event">';
  839. if(preg_grep("/item-movie/", $items)){
  840. $hideMenu .= '<li data-filter="item-movie" data-name="Movies" data-filter-on="false"><a class="js-filter-movie" href="javascript:void(0)">Movies</a></li>';
  841. }
  842. if(preg_grep("/item-season/", $items)){
  843. $hideMenu .= '<li data-filter="item-season" data-name="TV Shows" data-filter-on="false"><a class="js-filter-season" href="javascript:void(0)">Shows</a></li>';
  844. }
  845. if(preg_grep("/item-album/", $items)){
  846. $hideMenu .= '<li data-filter="item-album" data-name="Music Albums" data-filter-on="false"><a class="js-filter-album" href="javascript:void(0)">Music</a></li>';
  847. }
  848. $hideMenu .= '<li data-filter="item-all" data-name="Content" data-filter-on="false"><a class="js-filter-all" href="javascript:void(0)">All</a></li>';
  849. $hideMenu .= '</ul></div></div>';
  850. // If None Populate Empty Item
  851. if (!count($items)) {
  852. return '<div id="recentMedia" class="content-box box-shadow big-box"><h5 class="text-center">'.$header.'</h5><p class="text-center">No Media Found</p></div>';
  853. }else{
  854. $className = str_replace(' ', '', $header);
  855. return '<div id="recentMedia" class="content-box box-shadow big-box"><h5 id="recentContent-title" style="margin-bottom: -20px" class="text-center">'.$header.'</h5><div class="recentHeader inbox-pagination '.$className.'">'.$hideMenu.'</div><br/><br/><div class="recentItems-recent" data-name="'.$className.'">'.implode('',$items).'</div></div>'.($script?'<script>'.$script.'</script>':'');
  856. }
  857. }
  858. // Create Carousel
  859. function outputNowPlaying($header, $size, $type, $items, $script = false) {
  860. // If None Populate Empty Item
  861. if (!count($items)) {
  862. return '<div id=streamz></div>'.($script?'<script>'.$script.'</script>':'');
  863. }else{
  864. return '<div id=streamz><h5 class="zero-m big-box"><strong>'.$header.'</strong></h5>'.implode('',$items).'</div>'.($script?'<script>'.$script.'</script>':'');
  865. }
  866. }
  867. // Get Now Playing Streams From Emby
  868. function getEmbyStreams($size, $showNames, $role) {
  869. $address = qualifyURL(EMBYURL);
  870. $api = json_decode(@file_get_contents($address.'/Sessions?api_key='.EMBYTOKEN),true);
  871. if (!is_array($api)) { return 'Could not load!'; }
  872. $playingItems = array();
  873. foreach($api as $key => $value) {
  874. if (isset($value['NowPlayingItem'])) {
  875. $playingItems[] = resolveEmbyItem($address, EMBYTOKEN, $value['NowPlayingItem'], true, $showNames, $role, $value);
  876. }
  877. }
  878. return outputNowPlaying(translate('PLAYING_NOW_ON_EMBY'), $size, 'streams-emby', $playingItems, "
  879. setInterval(function() {
  880. $('<div></div>').load('ajax.php?a=emby-streams',function() {
  881. var element = $(this).find('[id]');
  882. var loadedID = element.attr('id');
  883. $('#'+loadedID).replaceWith(element);
  884. console.log('Loaded updated: '+loadedID);
  885. });
  886. }, 15000);
  887. ");
  888. }
  889. // Get Now Playing Streams From Plex
  890. function getPlexStreams($size, $showNames, $role){
  891. $address = qualifyURL(PLEXURL);
  892. // Perform API requests
  893. $api = @curl_get($address."/status/sessions?X-Plex-Token=".PLEXTOKEN);
  894. libxml_use_internal_errors(true);
  895. $api = simplexml_load_string($api);
  896. if (is_array($api) || is_object($api)){
  897. if (!$api->head->title){
  898. $getServer = simplexml_load_string(@curl_get($address."/?X-Plex-Token=".PLEXTOKEN));
  899. if (!$getServer) { return 'Could not load!'; }
  900. // Identify the local machine
  901. $gotServer = $getServer['machineIdentifier'];
  902. $items = array();
  903. foreach($api AS $child) {
  904. $items[] = resolvePlexItem($gotServer, PLEXTOKEN, $child, true, $showNames, $role);
  905. }
  906. return outputNowPlaying(translate('PLAYING_NOW_ON_PLEX')." ( ".count($items)." Streams )", $size, 'streams-plex', $items, "
  907. setInterval(function() {
  908. $('<div></div>').load('ajax.php?a=plex-streams',function() {
  909. var element = $(this).find('[id]');
  910. var loadedID = element.attr('id');
  911. $('#'+loadedID).replaceWith(element);
  912. console.log('Loaded updated: '+loadedID);
  913. });
  914. }, 15000);
  915. ");
  916. }else{
  917. writeLog("error", "PLEX STREAM ERROR: could not connect - check token - if HTTPS, is cert valid");
  918. }
  919. }else{
  920. writeLog("error", "PLEX STREAM ERROR: could not connect - check URL - if HTTPS, is cert valid");
  921. }
  922. }
  923. // Get Recent Content From Emby
  924. function getEmbyRecent($array) {
  925. $address = qualifyURL(EMBYURL);
  926. $header = translate('RECENT_CONTENT');
  927. // Currently Logged In User
  928. $username = false;
  929. if (isset($GLOBALS['USER'])) {
  930. $username = strtolower($GLOBALS['USER']->username);
  931. }
  932. // Get A User
  933. $userIds = json_decode(@file_get_contents($address.'/Users?api_key='.EMBYTOKEN),true);
  934. if (!is_array($userIds)) { return 'Could not load!'; }
  935. $showPlayed = true;
  936. foreach ($userIds as $value) { // Scan for admin user
  937. if (isset($value['Policy']) && isset($value['Policy']['IsAdministrator']) && $value['Policy']['IsAdministrator']) {
  938. $userId = $value['Id'];
  939. }
  940. if ($username && strtolower($value['Name']) == $username) {
  941. $userId = $value['Id'];
  942. $showPlayed = false;
  943. break;
  944. }
  945. }
  946. // Get the latest Items
  947. $latest = json_decode(file_get_contents($address.'/Users/'.$userId.'/Items/Latest?EnableImages=false&Limit='.EMBYRECENTITEMS.'&api_key='.EMBYTOKEN.($showPlayed?'':'&IsPlayed=false')),true);
  948. // For Each Item In Category
  949. $items = array();
  950. foreach ($latest as $k => $v) {
  951. $type = (string) $v['Type'];
  952. if(@$array[$type] == "true"){
  953. $items[] = resolveEmbyItem($address, EMBYTOKEN, $v, false, false, false);
  954. }
  955. }
  956. $array["movie"] = $array["Movie"];
  957. $array["season"] = $array["Episode"];
  958. $array["album"] = $array["MusicAlbum"];
  959. unset($array["Movie"]);
  960. unset($array["Episode"]);
  961. unset($array["MusicAlbum"]);
  962. unset($array["Series"]);
  963. return outputRecentAdded($header, $items, "", $array);
  964. }
  965. // Get Recent Content From Plex
  966. function getPlexRecent($array){
  967. $address = qualifyURL(PLEXURL);
  968. $header = translate('RECENT_CONTENT');
  969. // Perform Requests
  970. $api = @curl_get($address."/library/recentlyAdded?limit=".PLEXRECENTITEMS."&X-Plex-Token=".PLEXTOKEN);
  971. libxml_use_internal_errors(true);
  972. $api = simplexml_load_string($api);
  973. if (is_array($api) || is_object($api)){
  974. if (!$api->head->title){
  975. $getServer = simplexml_load_string(@curl_get($address."/?X-Plex-Token=".PLEXTOKEN));
  976. if (!$getServer) { return 'Could not load!'; }
  977. // Identify the local machine
  978. $gotServer = $getServer['machineIdentifier'];
  979. $items = array();
  980. foreach($api AS $child) {
  981. $type = (string) $child['type'];
  982. if($array[$type] == "true"){
  983. $items[] = resolvePlexItem($gotServer, PLEXTOKEN, $child, false, false, false);
  984. }
  985. }
  986. return outputRecentAdded($header, $items, "", $array);
  987. }else{
  988. writeLog("error", "PLEX RECENT-ITEMS ERROR: could not connect - check token - if HTTPS, is cert valid");
  989. }
  990. }else{
  991. writeLog("error", "PLEX RECENT-ITEMS ERROR: could not connect - check URL - if HTTPS, is cert valid");
  992. }
  993. }
  994. // Get Image From Emby
  995. function getEmbyImage() {
  996. $embyAddress = qualifyURL(EMBYURL);
  997. if (!file_exists('images/cache')) {
  998. mkdir('images/cache', 0777, true);
  999. }
  1000. $itemId = $_GET['img'];
  1001. $key = $_GET['key'];
  1002. $itemType = $_GET['type'];
  1003. $imgParams = array();
  1004. if (isset($_GET['height'])) { $imgParams['height'] = 'maxHeight='.$_GET['height']; }
  1005. if (isset($_GET['width'])) { $imgParams['width'] = 'maxWidth='.$_GET['width']; }
  1006. if(isset($itemId)) {
  1007. $image_src = $embyAddress . '/Items/'.$itemId.'/Images/'.$itemType.'?'.implode('&', $imgParams);
  1008. $cachefile = 'images/cache/'.$key.'.jpg';
  1009. $cachetime = 604800;
  1010. // Serve from the cache if it is younger than $cachetime
  1011. if (file_exists($cachefile) && time() - $cachetime < filemtime($cachefile)) {
  1012. header("Content-type: image/jpeg");
  1013. @readfile($cachefile);
  1014. exit;
  1015. }
  1016. ob_start(); // Start the output buffer
  1017. header('Content-type: image/jpeg');
  1018. @readfile($image_src);
  1019. // Cache the output to a file
  1020. $fp = fopen($cachefile, 'wb');
  1021. fwrite($fp, ob_get_contents());
  1022. fclose($fp);
  1023. ob_end_flush(); // Send the output to the browser
  1024. die();
  1025. } else {
  1026. debug_out('Invalid Request',1);
  1027. }
  1028. }
  1029. // Get Image From Plex
  1030. function getPlexImage() {
  1031. $plexAddress = qualifyURL(PLEXURL);
  1032. if (!file_exists('images/cache')) {
  1033. mkdir('images/cache', 0777, true);
  1034. }
  1035. $image_url = $_GET['img'];
  1036. $key = $_GET['key'];
  1037. $image_height = $_GET['height'];
  1038. $image_width = $_GET['width'];
  1039. if(isset($image_url) && isset($image_height) && isset($image_width)) {
  1040. $image_src = $plexAddress . '/photo/:/transcode?height='.$image_height.'&width='.$image_width.'&upscale=1&url=' . $image_url . '&X-Plex-Token=' . PLEXTOKEN;
  1041. $cachefile = 'images/cache/'.$key.'.jpg';
  1042. $cachetime = 604800;
  1043. // Serve from the cache if it is younger than $cachetime
  1044. if (file_exists($cachefile) && time() - $cachetime < filemtime($cachefile)) {
  1045. header("Content-type: image/jpeg");
  1046. @readfile($cachefile);
  1047. exit;
  1048. }
  1049. ob_start(); // Start the output buffer
  1050. header('Content-type: image/jpeg');
  1051. @readfile($image_src);
  1052. // Cache the output to a file
  1053. $fp = fopen($cachefile, 'wb');
  1054. fwrite($fp, ob_get_contents());
  1055. fclose($fp);
  1056. ob_end_flush(); // Send the output to the browser
  1057. die();
  1058. } else {
  1059. echo "Invalid Plex Request";
  1060. }
  1061. }
  1062. // Simplier access to class
  1063. function translate($string) {
  1064. if (isset($GLOBALS['language'])) {
  1065. return $GLOBALS['language']->translate($string);
  1066. } else {
  1067. return '!Translations Not Loaded!';
  1068. }
  1069. }
  1070. // Generate Random string
  1071. function randString($length = 10, $chars = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ') {
  1072. $tmp = '';
  1073. for ($i = 0; $i < $length; $i++) {
  1074. $tmp .= substr(str_shuffle($chars), 0, 1);
  1075. }
  1076. return $tmp;
  1077. }
  1078. // Create config file in the return syntax
  1079. function createConfig($array, $path = 'config/config.php', $nest = 0) {
  1080. // Define Initial Value
  1081. $output = array();
  1082. // Sort Items
  1083. ksort($array);
  1084. // Update the current config version
  1085. if (!$nest) {
  1086. // Inject Current Version
  1087. $output[] = "\t'CONFIG_VERSION' => '".(isset($array['apply_CONFIG_VERSION'])?$array['apply_CONFIG_VERSION']:INSTALLEDVERSION)."'";
  1088. }
  1089. unset($array['CONFIG_VERSION']);
  1090. unset($array['apply_CONFIG_VERSION']);
  1091. // Process Settings
  1092. foreach ($array as $k => $v) {
  1093. $allowCommit = true;
  1094. switch (gettype($v)) {
  1095. case 'boolean':
  1096. $item = ($v?'true':'false');
  1097. break;
  1098. case 'integer':
  1099. case 'double':
  1100. case 'integer':
  1101. case 'NULL':
  1102. $item = $v;
  1103. break;
  1104. case 'string':
  1105. $item = "'".str_replace(array('\\',"'"),array('\\\\',"\'"),$v)."'";
  1106. break;
  1107. case 'array':
  1108. $item = createConfig($v, false, $nest+1);
  1109. break;
  1110. default:
  1111. $allowCommit = false;
  1112. }
  1113. if($allowCommit) {
  1114. $output[] = str_repeat("\t",$nest+1)."'$k' => $item";
  1115. }
  1116. }
  1117. // Build output
  1118. $output = (!$nest?"<?php\nreturn ":'')."array(\n".implode(",\n",$output)."\n".str_repeat("\t",$nest).')'.(!$nest?';':'');
  1119. if (!$nest && $path) {
  1120. $pathDigest = pathinfo($path);
  1121. @mkdir($pathDigest['dirname'], 0770, true);
  1122. if (file_exists($path)) {
  1123. rename($path, $pathDigest['dirname'].'/'.$pathDigest['filename'].'.bak.php');
  1124. }
  1125. $file = fopen($path, 'w');
  1126. fwrite($file, $output);
  1127. fclose($file);
  1128. if (file_exists($path)) {
  1129. return true;
  1130. }
  1131. writeLog("error", "config was unable to write");
  1132. return false;
  1133. } else {
  1134. writeLog("success", "config was updated with new values");
  1135. return $output;
  1136. }
  1137. }
  1138. // Load a config file written in the return syntax
  1139. function loadConfig($path = 'config/config.php') {
  1140. // Adapted from http://stackoverflow.com/a/14173339/6810513
  1141. if (!is_file($path)) {
  1142. return null;
  1143. } else {
  1144. return (array) call_user_func(function() use($path) {
  1145. return include($path);
  1146. });
  1147. }
  1148. }
  1149. // Commit new values to the configuration
  1150. function updateConfig($new, $current = false) {
  1151. // Get config if not supplied
  1152. if ($current === false) {
  1153. $current = loadConfig();
  1154. } else if (is_string($current) && is_file($current)) {
  1155. $current = loadConfig($current);
  1156. }
  1157. // Inject Parts
  1158. foreach ($new as $k => $v) {
  1159. $current[$k] = $v;
  1160. }
  1161. // Return Create
  1162. return createConfig($current);
  1163. }
  1164. // Inject Defaults As Needed
  1165. function fillDefaultConfig($array, $path = 'config/configDefaults.php') {
  1166. if (is_string($path)) {
  1167. $loadedDefaults = loadConfig($path);
  1168. } else {
  1169. $loadedDefaults = $path;
  1170. }
  1171. return (is_array($loadedDefaults) ? fillDefaultConfig_recurse($array, $loadedDefaults) : false);
  1172. }
  1173. // support function for fillDefaultConfig()
  1174. function fillDefaultConfig_recurse($current, $defaults) {
  1175. foreach($defaults as $k => $v) {
  1176. if (!isset($current[$k])) {
  1177. $current[$k] = $v;
  1178. } else if (is_array($current[$k]) && is_array($v)) {
  1179. $current[$k] = fillDefaultConfig_recurse($current[$k], $v);
  1180. }
  1181. }
  1182. return $current;
  1183. };
  1184. // Define Scalar Variables (nest non-secular with underscores)
  1185. function defineConfig($array, $anyCase = true, $nest_prefix = false) {
  1186. foreach($array as $k => $v) {
  1187. if (is_scalar($v) && !defined($nest_prefix.$k)) {
  1188. define($nest_prefix.$k, $v, $anyCase);
  1189. } else if (is_array($v)) {
  1190. defineConfig($v, $anyCase, $nest_prefix.$k.'_');
  1191. }
  1192. }
  1193. }
  1194. // This function exists only because I am lazy
  1195. function configLazy($path = 'config/config.php') {
  1196. // Load config or default
  1197. if (file_exists($path)) {
  1198. $config = fillDefaultConfig(loadConfig($path));
  1199. } else {
  1200. $config = loadConfig('config/configDefaults.php');
  1201. }
  1202. if (is_array($config)) {
  1203. defineConfig($config);
  1204. }
  1205. return $config;
  1206. }
  1207. // Qualify URL
  1208. function qualifyURL($url) {
  1209. //local address?
  1210. if(substr($url, 0,1) == "/"){
  1211. if (isset($_SERVER['HTTPS']) && $_SERVER['HTTPS'] != 'off') {
  1212. $protocol = "https://";
  1213. } else {
  1214. $protocol = "http://";
  1215. }
  1216. $url = $protocol.getServer().$url;
  1217. }
  1218. // Get Digest
  1219. $digest = parse_url($url);
  1220. // http/https
  1221. if (!isset($digest['scheme'])) {
  1222. if (isset($digest['port']) && in_array($digest['port'], array(80,8080,8096,32400,7878,8989,8182,8081,6789))) {
  1223. $scheme = 'http';
  1224. } else {
  1225. $scheme = 'https';
  1226. }
  1227. } else {
  1228. $scheme = $digest['scheme'];
  1229. }
  1230. // Host
  1231. $host = (isset($digest['host'])?$digest['host']:'');
  1232. // Port
  1233. $port = (isset($digest['port'])?':'.$digest['port']:'');
  1234. // Path
  1235. $path = (isset($digest['path'])?$digest['path']:'');
  1236. // Output
  1237. return $scheme.'://'.$host.$port.$path;
  1238. }
  1239. // Function to be called at top of each to allow upgrading environment as the spec changes
  1240. function upgradeCheck() {
  1241. // Upgrade to 1.31
  1242. if (file_exists('homepageSettings.ini.php')) {
  1243. $databaseConfig = parse_ini_file('databaseLocation.ini.php', true);
  1244. $homepageConfig = parse_ini_file('homepageSettings.ini.php', true);
  1245. $databaseConfig = array_merge($databaseConfig, $homepageConfig);
  1246. $databaseData = '; <?php die("Access denied"); ?>' . "\r\n";
  1247. foreach($databaseConfig as $k => $v) {
  1248. if(substr($v, -1) == "/") : $v = rtrim($v, "/"); endif;
  1249. $databaseData .= $k . " = \"" . $v . "\"\r\n";
  1250. }
  1251. write_ini_file($databaseData, 'databaseLocation.ini.php');
  1252. unlink('homepageSettings.ini.php');
  1253. unset($databaseData);
  1254. unset($homepageConfig);
  1255. }
  1256. // Upgrade to 1.32
  1257. if (file_exists('databaseLocation.ini.php')) {
  1258. // Load Existing
  1259. $config = parse_ini_file('databaseLocation.ini.php', true);
  1260. // Refactor
  1261. $config['database_Location'] = preg_replace('/\/\/$/','/',$config['databaseLocation'].'/');
  1262. $config['user_home'] = $config['database_Location'].'users/';
  1263. unset($config['databaseLocation']);
  1264. // Turn Off Emby And Plex Recent
  1265. $config["embyURL"] = $config["embyURL"].(!empty($config["embyPort"])?':'.$config["embyPort"]:'');
  1266. unset($config["embyPort"]);
  1267. $config["plexURL"] = $config["plexURL"].(!empty($config["plexPort"])?':'.$config["plexPort"]:'');
  1268. unset($config["plexPort"]);
  1269. $config["nzbgetURL"] = $config["nzbgetURL"].(!empty($config["nzbgetPort"])?':'.$config["nzbgetPort"]:'');
  1270. unset($config["nzbgetPort"]);
  1271. $config["sabnzbdURL"] = $config["sabnzbdURL"].(!empty($config["sabnzbdPort"])?':'.$config["sabnzbdPort"]:'');
  1272. unset($config["sabnzbdPort"]);
  1273. $config["headphonesURL"] = $config["headphonesURL"].(!empty($config["headphonesPort"])?':'.$config["headphonesPort"]:'');
  1274. unset($config["headphonesPort"]);
  1275. // Write config file
  1276. $config['CONFIG_VERSION'] = '1.32';
  1277. copy('config/config.php', 'config/config['.date('Y-m-d_H-i-s').'][pre1.32].bak.php');
  1278. $createConfigSuccess = createConfig($config);
  1279. // Create new config
  1280. if ($createConfigSuccess) {
  1281. if (file_exists('config/config.php')) {
  1282. // Remove Old ini file
  1283. unlink('databaseLocation.ini.php');
  1284. } else {
  1285. debug_out('Something is not right here!');
  1286. }
  1287. } else {
  1288. debug_out('Couldn\'t create updated configuration.' ,1);
  1289. }
  1290. }
  1291. // Upgrade to 1.33
  1292. $config = loadConfig();
  1293. if (isset($config['database_Location']) && (!isset($config['CONFIG_VERSION']) || $config['CONFIG_VERSION'] < '1.33')) {
  1294. // Fix User Directory
  1295. $config['database_Location'] = preg_replace('/\/\/$/','/',$config['database_Location'].'/');
  1296. $config['user_home'] = $config['database_Location'].'users/';
  1297. unset($config['USER_HOME']);
  1298. // Backend auth merge
  1299. if (isset($config['authBackendPort']) && !isset(parse_url($config['authBackendHost'])['port'])) {
  1300. $config['authBackendHost'] .= ':'.$config['authBackendPort'];
  1301. }
  1302. unset($config['authBackendPort']);
  1303. // If auth is being used move it to embyURL as that is now used in auth functions
  1304. if ((isset($config['authType']) && $config['authType'] == 'true') && (isset($config['authBackendHost']) && $config['authBackendHost'] == 'true') && (isset($config['authBackend']) && in_array($config['authBackend'], array('emby_all','emby_local','emby_connect')))) {
  1305. $config['embyURL'] = $config['authBackendHost'];
  1306. }
  1307. // Upgrade database to latest version
  1308. updateSQLiteDB($config['database_Location'],'1.32');
  1309. // Update Version and Commit
  1310. $config['apply_CONFIG_VERSION'] = '1.33';
  1311. copy('config/config.php', 'config/config['.date('Y-m-d_H-i-s').'][1.32].bak.php');
  1312. $createConfigSuccess = createConfig($config);
  1313. unset($config);
  1314. }
  1315. // Upgrade to 1.34
  1316. $config = loadConfig();
  1317. if (isset($config['database_Location']) && (!isset($config['CONFIG_VERSION']) || $config['CONFIG_VERSION'] < '1.34')) {
  1318. // Upgrade database to latest version
  1319. updateSQLiteDB($config['database_Location'],'1.33');
  1320. // Update Version and Commit
  1321. $config['CONFIG_VERSION'] = '1.34';
  1322. copy('config/config.php', 'config/config['.date('Y-m-d_H-i-s').'][1.33].bak.php');
  1323. $createConfigSuccess = createConfig($config);
  1324. unset($config);
  1325. }
  1326. // Upgrade to 1.40
  1327. $config = loadConfig();
  1328. if (isset($config['database_Location']) && (!isset($config['CONFIG_VERSION']) || $config['CONFIG_VERSION'] < '1.40')) {
  1329. // Upgrade database to latest version
  1330. updateSQLiteDB($config['database_Location'],'1.38');
  1331. // Update Version and Commit
  1332. $config['CONFIG_VERSION'] = '1.40';
  1333. copy('config/config.php', 'config/config['.date('Y-m-d_H-i-s').'][1.38].bak.php');
  1334. $createConfigSuccess = createConfig($config);
  1335. unset($config);
  1336. }
  1337. // Upgrade to 1.50
  1338. $config = loadConfig();
  1339. if (isset($config['database_Location']) && (!isset($config['CONFIG_VERSION']) || $config['CONFIG_VERSION'] < '1.50')) {
  1340. // Upgrade database to latest version
  1341. updateSQLiteDB($config['database_Location'],'1.40');
  1342. // Update Version and Commit
  1343. $config['CONFIG_VERSION'] = '1.50';
  1344. copy('config/config.php', 'config/config['.date('Y-m-d_H-i-s').'][1.40].bak.php');
  1345. $createConfigSuccess = createConfig($config);
  1346. unset($config);
  1347. }
  1348. return true;
  1349. }
  1350. // Get OS from server
  1351. function getOS(){
  1352. if(PHP_SHLIB_SUFFIX == "dll"){
  1353. return "win";
  1354. }else{
  1355. return "nix";
  1356. }
  1357. }
  1358. //Get Error by Server OS
  1359. function getError($os, $error){
  1360. $ini = (!empty(php_ini_loaded_file()) ? php_ini_loaded_file() : "php.ini");
  1361. $ext = (!empty(ini_get('extension_dir')) ? "uncomment ;extension_dir = and make sure it says -> extension_dir = '".ini_get('extension_dir')."'" : "uncomment ;extension_dir = and add path to 'ext' to make it like extension_dir = 'C:\nginx\php\ext'");
  1362. $errors = array(
  1363. 'pdo_sqlite' => array(
  1364. 'win' => '<b>PDO:SQLite</b> not enabled, uncomment ;extension=php_pdo_sqlite.dll in the file php.ini | '.$ext,
  1365. 'nix' => '<b>PDO:SQLite</b> not enabled, PHP7 -> run sudo apt-get install php7.0-sqlite | PHP5 -> run sudo apt-get install php5-sqlite',
  1366. ),
  1367. 'sqlite3' => array(
  1368. 'win' => '<b>SQLite3</b> not enabled, uncomment ;extension=php_sqlite3.dll in the file php.ini | uncomment ;sqlite3.extension_dir = and add "ext" to make it sqlite3.extension_dir = ext',
  1369. 'nix' => '<b>SQLite3</b> not enabled, run sudo apt-get install php-sqlite3',
  1370. ),
  1371. 'curl' => array(
  1372. 'win' => '<b>cURL</b> not enabled, uncomment ;extension=php_curl.dll in the file php.ini | '.$ext,
  1373. 'nix' => '<b>cURL</b> not enabled, PHP7 -> sudo apt-get install php-curl or sudo apt-get install php7.0-curl | PHP5 -> run sudo apt-get install php5.6-curl',
  1374. ),
  1375. 'zip' => array(
  1376. 'win' => '<b>PHP Zip</b> not enabled, uncomment ;extension=php_zip.dll in the file php.ini, if that doesn\'t work remove that line',
  1377. 'nix' => '<b>PHP Zip</b> not enabled, PHP7 -> run sudo apt-get install php7.0-zip | PHP5 -> run sudo apt-get install php5.6-zip',
  1378. ),
  1379. );
  1380. return (isset($errors[$error][$os]) ? $errors[$error][$os] : 'No Error Info Found');
  1381. }
  1382. // Check if all software dependancies are met
  1383. function dependCheck() {
  1384. $output = array();
  1385. $i = 1;
  1386. if (!extension_loaded('pdo_sqlite')) { $output["Step $i"] = getError(getOS(),'pdo_sqlite'); $i++; }
  1387. if (!extension_loaded('curl')) { $output["Step $i"] = getError(getOS(),'curl'); $i++; }
  1388. if (!extension_loaded('zip')) { $output["Step $i"] = getError(getOS(),'zip'); $i++; }
  1389. //if (!extension_loaded('sqlite3')) { $output[] = getError(getOS(),'sqlite3'); }
  1390. if ($output) {
  1391. $output["Step $i"] = "<b>Restart PHP and/or Webserver to apply changes</b>"; $i++;
  1392. $output["Step $i"] = "<b>Please visit here to also check status of necessary components after you fix them: <a href='check.php'>check.php<a/></b>"; $i++;
  1393. debug_out($output,1);
  1394. }
  1395. return true;
  1396. }
  1397. // Process file uploads
  1398. function uploadFiles($path, $ext_mask = null) {
  1399. if (isset($_FILES) && count($_FILES)) {
  1400. require_once('class.uploader.php');
  1401. $uploader = new Uploader();
  1402. $data = $uploader->upload($_FILES['files'], array(
  1403. 'limit' => 10,
  1404. 'maxSize' => 10,
  1405. 'extensions' => $ext_mask,
  1406. 'required' => false,
  1407. 'uploadDir' => str_replace('//','/',$path.'/'),
  1408. 'title' => array('name'),
  1409. 'removeFiles' => true,
  1410. 'replace' => true,
  1411. ));
  1412. if($data['isComplete']){
  1413. $files = $data['data'];
  1414. writeLog("success", $files['metas'][0]['name']." was uploaded");
  1415. echo json_encode($files['metas'][0]['name']);
  1416. }
  1417. if($data['hasErrors']){
  1418. $errors = $data['errors'];
  1419. writeLog("error", $files['metas'][0]['name']." was not able to upload");
  1420. echo json_encode($errors);
  1421. }
  1422. } else {
  1423. writeLog("error", "image was not uploaded");
  1424. echo json_encode('No files submitted!');
  1425. }
  1426. }
  1427. // Process file uploads
  1428. function uploadAvatar($path, $ext_mask = null) {
  1429. if (isset($_FILES) && count($_FILES)) {
  1430. require_once('class.uploader.php');
  1431. $uploader = new Uploader();
  1432. $data = $uploader->upload($_FILES['files'], array(
  1433. 'limit' => 10,
  1434. 'maxSize' => 10,
  1435. 'extensions' => $ext_mask,
  1436. 'required' => false,
  1437. 'uploadDir' => str_replace('//','/',$path.'/'),
  1438. 'title' => array('name'),
  1439. 'removeFiles' => true,
  1440. 'replace' => true,
  1441. ));
  1442. if($data['isComplete']){
  1443. $files = $data['data'];
  1444. writeLog("success", $files['metas'][0]['name']." was uploaded");
  1445. echo json_encode($files['metas'][0]['name']);
  1446. }
  1447. if($data['hasErrors']){
  1448. $errors = $data['errors'];
  1449. writeLog("error", $files['metas'][0]['name']." was not able to upload");
  1450. echo json_encode($errors);
  1451. }
  1452. } else {
  1453. writeLog("error", "image was not uploaded");
  1454. echo json_encode('No files submitted!');
  1455. }
  1456. }
  1457. // Remove file
  1458. function removeFiles($path) {
  1459. if(is_file($path)) {
  1460. writeLog("success", "image was removed");
  1461. unlink($path);
  1462. } else {
  1463. writeLog("error", "image was not removed");
  1464. echo json_encode('No file specified for removal!');
  1465. }
  1466. }
  1467. // Lazy select options
  1468. function resolveSelectOptions($array, $selected = '', $multi = false) {
  1469. $output = array();
  1470. $selectedArr = ($multi?explode('|', $selected):array());
  1471. foreach ($array as $key => $value) {
  1472. if (is_array($value)) {
  1473. if (isset($value['optgroup'])) {
  1474. $output[] = '<optgroup label="'.$key.'">';
  1475. foreach($value['optgroup'] as $k => $v) {
  1476. $output[] = '<option value="'.$v['value'].'"'.($selected===$v['value']||in_array($v['value'],$selectedArr)?' selected':'').(isset($v['disabled']) && $v['disabled']?' disabled':'').'>'.$k.'</option>';
  1477. }
  1478. } else {
  1479. $output[] = '<option value="'.$value['value'].'"'.($selected===$value['value']||in_array($value['value'],$selectedArr)?' selected':'').(isset($value['disabled']) && $value['disabled']?' disabled':'').'>'.$key.'</option>';
  1480. }
  1481. } else {
  1482. $output[] = '<option value="'.$value.'"'.($selected===$value||in_array($value,$selectedArr)?' selected':'').'>'.$key.'</option>';
  1483. }
  1484. }
  1485. return implode('',$output);
  1486. }
  1487. // Check if user is allowed to continue
  1488. function qualifyUser($type, $errOnFail = false) {
  1489. if (!isset($GLOBALS['USER'])) {
  1490. require_once("user.php");
  1491. $GLOBALS['USER'] = new User('registration_callback');
  1492. }
  1493. if (is_bool($type)) {
  1494. if ($type === true) {
  1495. $authorized = ($GLOBALS['USER']->authenticated == true);
  1496. } else {
  1497. $authorized = true;
  1498. }
  1499. } elseif (is_string($type) || is_array($type)) {
  1500. if ($type !== 'false') {
  1501. if (!is_array($type)) {
  1502. $type = explode('|',$type);
  1503. }
  1504. $authorized = ($GLOBALS['USER']->authenticated && in_array($GLOBALS['USER']->role,$type));
  1505. } else {
  1506. $authorized = true;
  1507. }
  1508. } else {
  1509. debug_out('Invalid Syntax!',1);
  1510. }
  1511. if (!$authorized && $errOnFail) {
  1512. if ($GLOBALS['USER']->authenticated) {
  1513. header('Location: '.rtrim(dirname($_SERVER['SCRIPT_NAME']), '/').'/error.php?error=401');
  1514. echo '<script>window.location.href = \''.rtrim(dirname($_SERVER['SCRIPT_NAME']), '/').'/error.php?error=401\'</script>';
  1515. } else {
  1516. header('Location: '.rtrim(dirname($_SERVER['SCRIPT_NAME']), '/').'/error.php?error=999');
  1517. echo '<script>window.location.href = \''.rtrim(dirname($_SERVER['SCRIPT_NAME']), '/').'/error.php?error=999\'</script>';
  1518. }
  1519. debug_out('Not Authorized' ,1);
  1520. } else {
  1521. return $authorized;
  1522. }
  1523. }
  1524. // Build an (optionally) tabbed settings page.
  1525. function buildSettings($array) {
  1526. /*
  1527. array(
  1528. 'title' => '',
  1529. 'id' => '',
  1530. 'fields' => array( See buildField() ),
  1531. 'tabs' => array(
  1532. array(
  1533. 'title' => '',
  1534. 'id' => '',
  1535. 'image' => '',
  1536. 'fields' => array( See buildField() ),
  1537. ),
  1538. ),
  1539. );
  1540. */
  1541. $notifyExplode = explode("-", NOTIFYEFFECT);
  1542. $fieldFunc = function($fieldArr) {
  1543. $fields = '<div class="row">';
  1544. foreach($fieldArr as $key => $value) {
  1545. $isSingle = isset($value['type']);
  1546. if ($isSingle) { $value = array($value); }
  1547. $tmpField = '';
  1548. $sizeLg = max(floor(12/count($value)),2);
  1549. $sizeMd = max(floor(($isSingle?12:6)/count($value)),3);
  1550. foreach($value as $k => $v) {
  1551. $tmpField .= buildField($v, 12, $sizeMd, $sizeLg);
  1552. }
  1553. $fields .= ($isSingle?$tmpField:'<div class="row col-sm-12 content-form">'.$tmpField.'</div>');
  1554. }
  1555. $fields .= '</div>';
  1556. return $fields;
  1557. };
  1558. $fields = (isset($array['fields'])?$fieldFunc($array['fields']):'');
  1559. $tabSelectors = array();
  1560. $tabContent = array();
  1561. if (isset($array['tabs'])) {
  1562. foreach($array['tabs'] as $key => $value) {
  1563. $id = (isset($value['id'])?$value['id']:randString(32));
  1564. $tabSelectors[$key] = '<li class="apps'.($tabSelectors?'':' active').'"><a href="#tab-'.$id.'" data-toggle="tab" aria-expanded="true"><img style="height:40px; width:40px;" src="'.(isset($value['image'])?$value['image']:'images/organizr.png').'"></a></li>';
  1565. $tabContent[$key] = '<div class="tab-pane big-box fade'.($tabContent?'':' active in').'" id="tab-'.$id.'">'.$fieldFunc($value['fields']).'</div>';
  1566. }
  1567. }
  1568. $pageID = (isset($array['id'])?$array['id']:str_replace(array(' ','"',"'"),array('_'),strtolower($array['id'])));
  1569. $extraClick = ($pageID == 'appearance_settings' ? "$('#advanced_settings_form_submit').click();console.log('add theme settings');" : "");
  1570. return '
  1571. <div class="email-body">
  1572. <div class="email-header gray-bg">
  1573. <button type="button" class="btn btn-danger btn-sm waves close-button"><i class="fa fa-close"></i></button>
  1574. <button id="'.$pageID.'_form_submit" class="btn waves btn-labeled btn-success btn btn-sm text-uppercase waves-effect waves-float save-btn-form">
  1575. <span class="btn-label"><i class="fa fa-floppy-o"></i></span>Save
  1576. </button>
  1577. <h1>'.$array['title'].'</h1>
  1578. </div>
  1579. <div class="email-inner small-box">
  1580. <div class="email-inner-section">
  1581. <div class="small-box fade in" id="'.$pageID.'_frame">
  1582. <div class="col-lg-12">
  1583. '.(isset($array['customBeforeForm'])?$array['customBeforeForm']:'').'
  1584. <form class="content-form" name="'.$pageID.'" id="'.$pageID.'_form" onsubmit="return false;">
  1585. '.$fields.($tabContent?'
  1586. <div class="tabbable tabs-with-bg" id="'.$pageID.'_tabs">
  1587. <ul class="nav nav-tabs apps">
  1588. '.implode('', $tabSelectors).'
  1589. </ul>
  1590. <div class="clearfix"></div>
  1591. <div class="tab-content">
  1592. '.implode('', $tabContent).'
  1593. </div>
  1594. </div>':'').'
  1595. </form>
  1596. '.(isset($array['customAfterForm'])?$array['customAfterForm']:'').'
  1597. </div>
  1598. </div>
  1599. </div>
  1600. </div>
  1601. </div>
  1602. <script>
  1603. $(document).ready(function() {
  1604. $(\'#'.$pageID.'_form\').find(\'input, select, textarea\').on(\'change\', function() { $(this).attr(\'data-changed\', \'true\'); });
  1605. var '.$pageID.'Validate = function() { if (this.value && !RegExp(\'^\'+this.pattern+\'$\').test(this.value)) { $(this).addClass(\'invalid\'); } else { $(this).removeClass(\'invalid\'); } };
  1606. $(\'#'.$pageID.'_form\').find(\'input[pattern]\').each('.$pageID.'Validate).on(\'keyup\', '.$pageID.'Validate);
  1607. $(\'#'.$pageID.'_form\').find(\'select[multiple]\').on(\'change click\', function() { $(this).attr(\'data-changed\', \'true\'); });
  1608. $(\'#'.$pageID.'_form_submit\').on(\'click\', function () {
  1609. var newVals = {};
  1610. var hasVals = false;
  1611. var errorFields = [];
  1612. $(\'#'.$pageID.'_form\').find(\'[data-changed=true][name]\').each(function() {
  1613. hasVals = true;
  1614. if (this.type == \'checkbox\') {
  1615. newVals[this.name] = this.checked;
  1616. } else if ($(this).hasClass(\'summernote\')) {
  1617. newVals[$(this).attr(\'name\')] = $(this).siblings(\'.note-editor\').find(\'.panel-body\').html();
  1618. } else {
  1619. if (this.value && this.pattern && !RegExp(\'^\'+this.pattern+\'$\').test(this.value)) { errorFields.push(this.name); }
  1620. var fieldVal = $(this).val();
  1621. if (typeof fieldVal == \'object\') {
  1622. if (typeof fieldVal.join == \'function\') {
  1623. fieldVal = fieldVal.join(\'|\');
  1624. } else {
  1625. fieldVal = JSON.stringify(fieldVal);
  1626. }
  1627. }
  1628. newVals[this.name] = fieldVal;
  1629. }
  1630. });
  1631. if (errorFields.length) {
  1632. parent.notify(\'Fields have errors: \'+errorFields.join(\', \')+\'!\', \'bullhorn\', \'error\', 5000, \''.$notifyExplode[0].'\', \''.$notifyExplode[1].'\');
  1633. } else if (hasVals) {
  1634. console.log(newVals);
  1635. ajax_request(\'POST\', \''.(isset($array['submitAction'])?$array['submitAction']:'update-config').'\', newVals, function(data, code) {
  1636. $(\'#'.$pageID.'_form\').find(\'[data-changed=true][name]\').removeAttr(\'data-changed\');
  1637. });
  1638. '.$extraClick.'
  1639. } else {
  1640. parent.notify(\'Nothing to update!\', \'bullhorn\', \'error\', 5000, \''.$notifyExplode[0].'\', \''.$notifyExplode[1].'\');
  1641. }
  1642. return false;
  1643. });
  1644. '.(isset($array['onready'])?$array['onready']:'').'
  1645. });
  1646. </script>
  1647. ';
  1648. }
  1649. // Build Settings Fields
  1650. function buildField($params, $sizeSm = 12, $sizeMd = 12, $sizeLg = 12) {
  1651. /*
  1652. array(
  1653. 'type' => '',
  1654. 'placeholder' => '',
  1655. 'label' => '',
  1656. 'labelTranslate' => '',
  1657. 'assist' => '',
  1658. 'name' => '',
  1659. 'pattern' => '',
  1660. 'options' => array( // For SELECT only
  1661. 'Display' => 'value',
  1662. ),
  1663. )
  1664. */
  1665. // Tags
  1666. $tags = array();
  1667. foreach(array('placeholder','style','disabled','readonly','pattern','min','max','required','onkeypress','onchange','onfocus','onleave','href','onclick') as $value) {
  1668. if (isset($params[$value])) {
  1669. if (is_string($params[$value])) { $tags[] = $value.'="'.$params[$value].'"';
  1670. } else if ($params[$value] === true) { $tags[] = $value; }
  1671. }
  1672. }
  1673. $format = (isset($params['format']) && in_array($params['format'],array(false,'colour','color'))?$params['format']:false);
  1674. $name = (isset($params['name'])?$params['name']:(isset($params['id'])?$params['id']:''));
  1675. $id = (isset($params['id'])?$params['id']:(isset($params['name'])?$params['name'].'_id':randString(32)));
  1676. $val = (isset($params['value'])?$params['value']:'');
  1677. $class = (isset($params['class'])?' '.$params['class']:'');
  1678. $wrapClass = (isset($params['wrapClass'])?$params['wrapClass']:'form-content');
  1679. $assist = (isset($params['assist'])?' - i.e. '.$params['assist']:'');
  1680. $label = (isset($params['labelTranslate'])?translate($params['labelTranslate']):(isset($params['label'])?$params['label']:''));
  1681. $labelOut = '<p class="help-text">'.$label.$assist.'</p>';
  1682. // Field Design
  1683. switch ($params['type']) {
  1684. case 'text':
  1685. case 'number':
  1686. case 'password':
  1687. $field = '<input id="'.$id.'" name="'.$name.'" type="'.$params['type'].'" class="form-control material input-sm'.$class.'" '.implode(' ',$tags).' autocorrect="off" autocapitalize="off" value="'.$val.'">';
  1688. break;
  1689. case 'select':
  1690. case 'dropdown':
  1691. $field = '<select id="'.$id.'" name="'.$name.'" class="form-control material input-sm" '.implode(' ',$tags).'>'.resolveSelectOptions($params['options'], $val).'</select>';
  1692. break;
  1693. case 'select-multi':
  1694. case 'dropdown-multi':
  1695. $field = '<select id="'.$id.'" name="'.$name.'" class="form-control input-sm" '.implode(' ',$tags).' multiple="multiple">'.resolveSelectOptions($params['options'], $val, true).'</select>';
  1696. break;
  1697. case 'check':
  1698. case 'checkbox':
  1699. case 'toggle':
  1700. $checked = ((is_bool($val) && $val) || trim($val) === 'true'?' checked':'');
  1701. $colour = (isset($params['colour'])?$params['colour']:'success');
  1702. $labelOut = '<label for="'.$id.'"></label>'.$label;
  1703. $field = '<input id="'.$id.'" name="'.$name.'" type="checkbox" class="switcher switcher-'.$colour.' '.$class.'" '.implode(' ',$tags).' data-value="'.$val.'"'.$checked.'>';
  1704. break;
  1705. case 'radio':
  1706. $labelOut = '';
  1707. $checked = ((is_bool($val) && $val) || ($val && trim($val) !== 'false')?' checked':'');
  1708. $bType = (isset($params['buttonType'])?$params['buttonType']:'success');
  1709. $field = '<div class="radio radio-'.$bType.'"><input id="'.$id.'" name="'.$name.'" type="radio" class="'.$class.'" '.implode(' ',$tags).' value="'.$val.'"'.$checked.'><label for="'.$id.'">'.$label.'</label></div>';
  1710. break;
  1711. case 'date':
  1712. $field = 'Unsupported, planned.';
  1713. break;
  1714. case 'hidden':
  1715. return '<input id="'.$id.'" name="'.$name.'" type="hidden" class="'.$class.'" '.implode(' ',$tags).' value="'.$val.'">';
  1716. break;
  1717. case 'header':
  1718. $labelOut = '';
  1719. $headType = (isset($params['value'])?$params['value']:3);
  1720. $field = '<h'.$headType.' class="'.$class.'" '.implode(' ',$tags).'>'.$label.'</h'.$headType.'>';
  1721. break;
  1722. case 'button':
  1723. $labelOut = '';
  1724. $icon = (isset($params['icon'])?$params['icon']:'flask');
  1725. $bType = (isset($params['buttonType'])?$params['buttonType']:'success');
  1726. $bDropdown = (isset($params['buttonDrop'])?$params['buttonDrop']:'');
  1727. $field = ($bDropdown?'<div class="btn-group">':'').'<button id="'.$id.'" type="button" class="btn waves btn-labeled btn-'.$bType.' btn-sm text-uppercase waves-effect waves-float'.$class.''.($bDropdown?' dropdown-toggle" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false"':'"').' '.implode(' ',$tags).'><span class="btn-label"><i class="fa fa-'.$icon.'"></i></span><span class="btn-text">'.$label.'</span></button>'.($bDropdown?$bDropdown.'</div>':'');
  1728. break;
  1729. case 'textarea':
  1730. $rows = (isset($params['rows'])?$params['rows']:5);
  1731. $field = '<textarea id="'.$id.'" name="'.$name.'" class="form-control'.$class.'" rows="'.$rows.'" '.implode(' ',$tags).'>'.$val.'</textarea>';
  1732. break;
  1733. case 'custom':
  1734. // Settings
  1735. $settings = array(
  1736. '$id' => $id,
  1737. '$name' => $name,
  1738. '$val' => $val,
  1739. '$label' => $label,
  1740. '$labelOut' => $labelOut,
  1741. );
  1742. // Get HTML
  1743. $html = (isset($params['html'])?$params['html']:'Nothing Specified!');
  1744. // If LabelOut is in html dont print it twice
  1745. $labelOut = (strpos($html,'$label')!==false?'':$labelOut);
  1746. // Replace variables in settings
  1747. $html = preg_replace_callback('/\$\w+\b/', function ($match) use ($settings) { return (isset($settings[$match[0]])?$settings[$match[0]]:'{'.$match[0].' is undefined}'); }, $html);
  1748. // Build Field
  1749. $field = '<div id="'.$id.'_html" class="custom-field">'.$html.'</div>';
  1750. break;
  1751. case 'space':
  1752. $labelOut = '';
  1753. $field = str_repeat('<br>', (isset($params['value'])?$params['value']:1));
  1754. break;
  1755. default:
  1756. $field = 'Unsupported field type';
  1757. break;
  1758. }
  1759. // Field Formats
  1760. switch ($format) {
  1761. case 'colour': // Fuckin Eh, Canada!
  1762. case 'color':
  1763. $labelBef = '<center>'.$label.'</center>';
  1764. $wrapClass = 'gray-bg colour-field';
  1765. $labelAft = '';
  1766. $field = str_replace(' material input-sm','',$field);
  1767. break;
  1768. default:
  1769. $labelBef = '';
  1770. $labelAft = $labelOut;
  1771. }
  1772. return '<div class="'.$wrapClass.' col-sm-'.$sizeSm.' col-md-'.$sizeMd.' col-lg-'.$sizeLg.'">'.$labelBef.$field.$labelAft.'</div>';
  1773. }
  1774. // Tab Settings Generation
  1775. function printTabRow($data) {
  1776. $hidden = false;
  1777. if ($data===false) {
  1778. $hidden = true;
  1779. $data = array( // New Tab Defaults
  1780. 'id' => 'new',
  1781. 'name' => '',
  1782. 'url' => '',
  1783. 'icon' => 'fa-diamond',
  1784. 'iconurl' => '',
  1785. 'active' => 'true',
  1786. 'user' => 'true',
  1787. 'guest' => 'true',
  1788. 'window' => 'false',
  1789. 'splash' => 'true',
  1790. 'ping' => 'false',
  1791. 'ping_url' => '',
  1792. 'defaultz' => '',
  1793. );
  1794. }
  1795. $image = '<span style="font: normal normal normal 30px/1 FontAwesome;" class="fa fa-hand-paper-o"></span>';
  1796. $output = '
  1797. <li id="tab-'.$data['id'].'" class="list-group-item" style="position: relative; left: 0px; top: 0px; '.($hidden?' display: none;':'').'">
  1798. <tab class="content-form form-inline">
  1799. <div class="row">
  1800. '.buildField(array(
  1801. 'type' => 'custom',
  1802. 'html' => '<div class="action-btns tabIconView"><a style="margin-left: 0px">'.($data['iconurl']?'<img src="'.$data['iconurl'].'" height="30" width="30">':'<span style="font: normal normal normal 30px/1 FontAwesome;" class="fa '.($data['icon']?$data['icon']:'hand-paper-o').'"></span>').'</a></div>',
  1803. ),12,1,1).'
  1804. '.buildField(array(
  1805. 'type' => 'hidden',
  1806. 'id' => 'tab-'.$data['id'].'-id',
  1807. 'name' => 'id['.$data['id'].']',
  1808. 'value' => $data['id'],
  1809. ),12,2,1).'
  1810. '.buildField(array(
  1811. 'type' => 'text',
  1812. 'id' => 'tab-'.$data['id'].'-name',
  1813. 'name' => 'name['.$data['id'].']',
  1814. 'required' => true,
  1815. 'placeholder' => 'Organizr Homepage',
  1816. 'labelTranslate' => 'TAB_NAME',
  1817. 'value' => $data['name'],
  1818. 'class' => 'darkBold',
  1819. ),12,2,1).'
  1820. '.buildField(array(
  1821. 'type' => 'text',
  1822. 'id' => 'tab-'.$data['id'].'-url',
  1823. 'name' => 'url['.$data['id'].']',
  1824. 'required' => true,
  1825. 'placeholder' => 'homepage.php',
  1826. 'labelTranslate' => 'TAB_URL',
  1827. 'value' => $data['url'],
  1828. 'class' => 'darkBold',
  1829. ),12,2,2).'
  1830. '.buildField(array(
  1831. 'type' => 'text',
  1832. 'id' => 'tab-'.$data['id'].'-iconurl',
  1833. 'name' => 'iconurl['.$data['id'].']',
  1834. 'placeholder' => 'images/organizr.png',
  1835. 'labelTranslate' => 'ICON_URL',
  1836. 'value' => $data['iconurl'],
  1837. 'class' => 'darkBold',
  1838. ),12,2,1).'
  1839. '.buildField(array(
  1840. 'type' => 'custom',
  1841. 'id' => 'tab-'.$data['id'].'-icon',
  1842. 'name' => 'icon['.$data['id'].']',
  1843. 'html' => '- '.translate('OR').' - <div class="input-group"><input data-placement="bottomRight" class="form-control material icp-auto'.($hidden?'-pend':'').'" id="$id" name="$name" value="$val" type="text" /><span class="input-group-addon"></span></div>',
  1844. 'value' => $data['icon'],
  1845. ),12,1,1).'
  1846. '.buildField(array(
  1847. 'type' => 'text',
  1848. 'id' => 'tab-'.$data['id'].'-ping_url',
  1849. 'name' => 'ping_url['.$data['id'].']',
  1850. 'placeholder' => 'host:port',
  1851. 'labelTranslate' => 'PING_URL',
  1852. 'value' => $data['ping_url'],
  1853. 'class' => 'darkBold',
  1854. ),12,2,1).'
  1855. '.buildField(array(
  1856. 'type' => 'radio',
  1857. 'labelTranslate' => 'DEFAULT',
  1858. 'name' => 'defaultz['.$data['id'].']',
  1859. 'value' => $data['defaultz'],
  1860. 'onclick' => "$('[type=radio][id!=\''+this.id+'\']').each(function() { this.checked=false; });",
  1861. ),12,1,1).'
  1862. '.buildField(array(
  1863. 'type' => 'button',
  1864. 'icon' => 'chevron-down',
  1865. 'buttonType' => 'success',
  1866. 'labelTranslate' => 'MORE',
  1867. 'onclick' => "$(this).parent().parent().parent().find('.slideInUp').toggle()",
  1868. 'class' => 'toggleTabExtra',
  1869. ),12,1,1).'
  1870. '.buildField(array(
  1871. 'type' => 'button',
  1872. 'icon' => 'trash',
  1873. 'buttonType' => 'danger',
  1874. 'labelTranslate' => 'REMOVE',
  1875. 'onclick' => "$(this).parents('li').remove();",
  1876. ),12,1,1).'</div><div id = "tab-'.$data['id'].'-row" class = "row animated slideInUp" style = "display:none;" ><div></div>
  1877. '.buildField(array(
  1878. 'type' => 'checkbox',
  1879. 'labelTranslate' => 'ACTIVE',
  1880. 'name' => 'active['.$data['id'].']',
  1881. 'value' => $data['active'],
  1882. ),12,1,1).'
  1883. '.buildField(array(
  1884. 'type' => 'checkbox',
  1885. 'labelTranslate' => 'USER',
  1886. 'colour' => 'primary',
  1887. 'name' => 'user['.$data['id'].']',
  1888. 'value' => $data['user'],
  1889. ),12,1,1).'
  1890. '.buildField(array(
  1891. 'type' => 'checkbox',
  1892. 'labelTranslate' => 'GUEST',
  1893. 'colour' => 'warning',
  1894. 'name' => 'guest['.$data['id'].']',
  1895. 'value' => $data['guest'],
  1896. ),12,1,1).'
  1897. '.buildField(array(
  1898. 'type' => 'checkbox',
  1899. 'labelTranslate' => 'NO_IFRAME',
  1900. 'colour' => 'danger',
  1901. 'name' => 'window['.$data['id'].']',
  1902. 'value' => $data['window'],
  1903. ),12,1,1).'
  1904. '.buildField(array(
  1905. 'type' => 'checkbox',
  1906. 'labelTranslate' => 'SPLASH',
  1907. 'colour' => 'success',
  1908. 'name' => 'splash['.$data['id'].']',
  1909. 'value' => $data['splash'],
  1910. ),12,1,1).'
  1911. '.buildField(array(
  1912. 'type' => 'checkbox',
  1913. 'labelTranslate' => 'PING',
  1914. 'colour' => 'success',
  1915. 'name' => 'ping['.$data['id'].']',
  1916. 'value' => $data['ping'],
  1917. ),12,1,1).'
  1918. </div>
  1919. </tab>
  1920. </li>
  1921. ';
  1922. return $output;
  1923. }
  1924. // Timezone array
  1925. function timezoneOptions() {
  1926. $output = array();
  1927. $timezones = array();
  1928. $regions = array(
  1929. 'Africa' => DateTimeZone::AFRICA,
  1930. 'America' => DateTimeZone::AMERICA,
  1931. 'Antarctica' => DateTimeZone::ANTARCTICA,
  1932. 'Arctic' => DateTimeZone::ARCTIC,
  1933. 'Asia' => DateTimeZone::ASIA,
  1934. 'Atlantic' => DateTimeZone::ATLANTIC,
  1935. 'Australia' => DateTimeZone::AUSTRALIA,
  1936. 'Europe' => DateTimeZone::EUROPE,
  1937. 'Indian' => DateTimeZone::INDIAN,
  1938. 'Pacific' => DateTimeZone::PACIFIC
  1939. );
  1940. foreach ($regions as $name => $mask) {
  1941. $zones = DateTimeZone::listIdentifiers($mask);
  1942. foreach($zones as $timezone) {
  1943. $time = new DateTime(NULL, new DateTimeZone($timezone));
  1944. $ampm = $time->format('H') > 12 ? ' ('. $time->format('g:i a'). ')' : '';
  1945. $output[$name]['optgroup'][substr($timezone, strlen($name) + 1) . ' - ' . $time->format('H:i') . $ampm]['value'] = $timezone;
  1946. }
  1947. }
  1948. return $output;
  1949. }
  1950. // Build Database
  1951. function createSQLiteDB($path = false) {
  1952. if ($path === false) {
  1953. if (DATABASE_LOCATION){
  1954. $path = DATABASE_LOCATION;
  1955. } else {
  1956. debug_out('No Path Specified!');
  1957. }
  1958. }
  1959. if (!is_file($path.'users.db') || filesize($path.'users.db') <= 0) {
  1960. if (!isset($GLOBALS['file_db'])) {
  1961. $GLOBALS['file_db'] = new PDO('sqlite:'.$path.'users.db');
  1962. $GLOBALS['file_db']->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  1963. }
  1964. // Create Users
  1965. $users = $GLOBALS['file_db']->query('CREATE TABLE `users` (
  1966. `id` INTEGER PRIMARY KEY AUTOINCREMENT UNIQUE,
  1967. `username` TEXT UNIQUE,
  1968. `password` TEXT,
  1969. `email` TEXT,
  1970. `token` TEXT,
  1971. `role` TEXT,
  1972. `active` TEXT,
  1973. `last` TEXT,
  1974. `auth_service` TEXT DEFAULT \'internal\'
  1975. );');
  1976. // Create Tabs
  1977. $tabs = $GLOBALS['file_db']->query('CREATE TABLE `tabs` (
  1978. `id` INTEGER PRIMARY KEY AUTOINCREMENT UNIQUE,
  1979. `order` INTEGER,
  1980. `users_id` INTEGER,
  1981. `name` TEXT,
  1982. `url` TEXT,
  1983. `defaultz` TEXT,
  1984. `active` TEXT,
  1985. `user` TEXT,
  1986. `guest` TEXT,
  1987. `icon` TEXT,
  1988. `iconurl` TEXT,
  1989. `window` TEXT,
  1990. `splash` TEXT,
  1991. `ping` TEXT,
  1992. `ping_url` TEXT
  1993. );');
  1994. // Create Options
  1995. $options = $GLOBALS['file_db']->query('CREATE TABLE `options` (
  1996. `id` INTEGER PRIMARY KEY AUTOINCREMENT UNIQUE,
  1997. `users_id` INTEGER UNIQUE,
  1998. `title` TEXT UNIQUE,
  1999. `topbar` TEXT,
  2000. `bottombar` TEXT,
  2001. `sidebar` TEXT,
  2002. `hoverbg` TEXT,
  2003. `topbartext` TEXT,
  2004. `activetabBG` TEXT,
  2005. `activetabicon` TEXT,
  2006. `activetabtext` TEXT,
  2007. `inactiveicon` TEXT,
  2008. `inactivetext` TEXT,
  2009. `loading` TEXT,
  2010. `hovertext` TEXT
  2011. );');
  2012. // Create Invites
  2013. $invites = $GLOBALS['file_db']->query('CREATE TABLE `invites` (
  2014. `id` INTEGER PRIMARY KEY AUTOINCREMENT UNIQUE,
  2015. `code` TEXT UNIQUE,
  2016. `date` TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
  2017. `email` TEXT,
  2018. `username` TEXT,
  2019. `dateused` TIMESTAMP,
  2020. `usedby` TEXT,
  2021. `ip` TEXT,
  2022. `valid` TEXT
  2023. );');
  2024. writeLog("success", "database created/saved");
  2025. return $users && $tabs && $options && $invites;
  2026. } else {
  2027. writeLog("error", "database was unable to be created/saved");
  2028. return false;
  2029. }
  2030. }
  2031. // Upgrade Database
  2032. function updateSQLiteDB($db_path = false, $oldVerNum = false) {
  2033. if (!$db_path) {
  2034. if (defined('DATABASE_LOCATION')) {
  2035. $db_path = DATABASE_LOCATION;
  2036. } else {
  2037. debug_out('No Path Specified',1);
  2038. }
  2039. }
  2040. if (!isset($GLOBALS['file_db'])) {
  2041. $GLOBALS['file_db'] = new PDO('sqlite:'.$db_path.'users.db');
  2042. $GLOBALS['file_db']->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  2043. }
  2044. // Cache current DB
  2045. $cache = array();
  2046. foreach($GLOBALS['file_db']->query('SELECT name FROM sqlite_master WHERE type="table";') as $table) {
  2047. foreach($GLOBALS['file_db']->query('SELECT * FROM '.$table['name'].';') as $key => $row) {
  2048. foreach($row as $k => $v) {
  2049. if (is_string($k)) {
  2050. $cache[$table['name']][$key][$k] = $v;
  2051. }
  2052. }
  2053. }
  2054. }
  2055. // Remove Current Database
  2056. $GLOBALS['file_db'] = null;
  2057. $pathDigest = pathinfo($db_path.'users.db');
  2058. if (file_exists($db_path.'users.db')) {
  2059. rename($db_path.'users.db', $pathDigest['dirname'].'/'.$pathDigest['filename'].'['.date('Y-m-d_H-i-s').']'.($oldVerNum?'['.$oldVerNum.']':'').'.bak.db');
  2060. }
  2061. // Create New Database
  2062. $success = createSQLiteDB($db_path);
  2063. // Restore Items
  2064. if ($success) {
  2065. foreach($cache as $table => $tableData) {
  2066. if ($tableData) {
  2067. $queryBase = 'INSERT INTO '.$table.' (`'.implode('`,`',array_keys(current($tableData))).'`) values ';
  2068. $insertValues = array();
  2069. reset($tableData);
  2070. foreach($tableData as $key => $value) {
  2071. $insertValues[] = '('.implode(',',array_map(function($d) {
  2072. return (isset($d)?$GLOBALS['file_db']->quote($d):'null');
  2073. }, $value)).')';
  2074. }
  2075. $GLOBALS['file_db']->query($queryBase.implode(',',$insertValues).';');
  2076. }
  2077. }
  2078. writeLog("success", "database values have been updated");
  2079. return true;
  2080. } else {
  2081. writeLog("error", "database values unable to be updated");
  2082. return false;
  2083. }
  2084. }
  2085. // Commit colours to database
  2086. function updateDBOptions($values) {
  2087. if (!isset($GLOBALS['file_db'])) {
  2088. $GLOBALS['file_db'] = new PDO('sqlite:'.DATABASE_LOCATION.'users.db');
  2089. $GLOBALS['file_db']->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  2090. }
  2091. // Commit new values to database
  2092. if ($GLOBALS['file_db']->query('UPDATE options SET '.implode(',',array_map(function($d, $k) {
  2093. return '`'.$k.'` = '.(isset($d)?"'".addslashes($d)."'":'null');
  2094. }, $values, array_keys($values))).';')->rowCount()) {
  2095. return true;
  2096. } else if ($GLOBALS['file_db']->query('INSERT OR IGNORE INTO options (`'.implode('`,`',array_keys($values)).'`) VALUES (\''.implode("','",$values).'\');')->rowCount()) {
  2097. writeLog("success", "database values for options table have been updated");
  2098. return true;
  2099. } else {
  2100. writeLog("error", "database values for options table unable to be updated");
  2101. return false;
  2102. }
  2103. }
  2104. // Send AJAX notification
  2105. function sendNotification($success, $message = false, $send = true) {
  2106. $notifyExplode = explode("-", NOTIFYEFFECT);
  2107. if ($success) {
  2108. $msg = array(
  2109. 'html' => ($message?''.$message:'<strong>'.translate("SETTINGS_SAVED").'</strong>'),
  2110. 'icon' => 'floppy-o',
  2111. 'type' => 'success',
  2112. 'length' => '5000',
  2113. 'layout' => $notifyExplode[0],
  2114. 'effect' => $notifyExplode[1],
  2115. );
  2116. } else {
  2117. $msg = array(
  2118. 'html' => ($message?''.$message:'<strong>'.translate("SETTINGS_NOT_SAVED").'</strong>'),
  2119. 'icon' => 'floppy-o',
  2120. 'type' => 'failed',
  2121. 'length' => '5000',
  2122. 'layout' => $notifyExplode[0],
  2123. 'effect' => $notifyExplode[1],
  2124. );
  2125. }
  2126. // Send and kill script?
  2127. if ($send) {
  2128. header('Content-Type: application/json');
  2129. echo json_encode(array('notify'=>$msg));
  2130. die();
  2131. }
  2132. return $msg;
  2133. }
  2134. // Load colours from the database
  2135. function loadAppearance() {
  2136. // Defaults
  2137. $defaults = array(
  2138. 'title' => 'Organizr',
  2139. 'topbartext' => '#66D9EF',
  2140. 'topbar' => '#333333',
  2141. 'bottombar' => '#333333',
  2142. 'sidebar' => '#393939',
  2143. 'hoverbg' => '#AD80FD',
  2144. 'activetabBG' => '#F92671',
  2145. 'activetabicon' => '#FFFFFF',
  2146. 'activetabtext' => '#FFFFFF',
  2147. 'inactiveicon' => '#66D9EF',
  2148. 'inactivetext' => '#66D9EF',
  2149. 'loading' => '#66D9EF',
  2150. 'hovertext' => '#000000',
  2151. );
  2152. if (DATABASE_LOCATION) {
  2153. if(is_file(DATABASE_LOCATION.'users.db') && filesize(DATABASE_LOCATION.'users.db') > 0){
  2154. if (!isset($GLOBALS['file_db'])) {
  2155. $GLOBALS['file_db'] = new PDO('sqlite:'.DATABASE_LOCATION.'users.db');
  2156. $GLOBALS['file_db']->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  2157. }
  2158. // Database Lookup
  2159. $options = $GLOBALS['file_db']->query('SELECT * FROM options');
  2160. // Replace defaults with filled options
  2161. foreach($options as $row) {
  2162. foreach($defaults as $key => $value) {
  2163. if (isset($row[$key]) && $row[$key]) {
  2164. $defaults[$key] = $row[$key];
  2165. }
  2166. }
  2167. }
  2168. }
  2169. }
  2170. // Return the Results
  2171. return $defaults;
  2172. }
  2173. // Delete Database
  2174. function deleteDatabase() {
  2175. unset($_COOKIE['Organizr']);
  2176. setcookie('Organizr', '', time() - 3600, '/');
  2177. unset($_COOKIE['OrganizrU']);
  2178. setcookie('OrganizrU', '', time() - 3600, '/');
  2179. $GLOBALS['file_db'] = null;
  2180. unlink(DATABASE_LOCATION.'users.db');
  2181. foreach(glob(substr_replace($userdirpath, "", -1).'/*') as $file) {
  2182. if(is_dir($file)) {
  2183. rmdir($file);
  2184. } elseif (!is_dir($file)) {
  2185. unlink($file);
  2186. }
  2187. }
  2188. rmdir($userdirpath);
  2189. writeLog("success", "database has been deleted");
  2190. return true;
  2191. }
  2192. // Upgrade the installation
  2193. function upgradeInstall($branch = 'master') {
  2194. function downloadFile($url, $path){
  2195. ini_set('max_execution_time',0);
  2196. $folderPath = "upgrade/";
  2197. if(!mkdir($folderPath)){
  2198. writeLog("error", "organizr could not create upgrade folder");
  2199. }
  2200. $newfname = $folderPath . $path;
  2201. $file = fopen ($url, 'rb');
  2202. if ($file) {
  2203. $newf = fopen ($newfname, 'wb');
  2204. if ($newf) {
  2205. while(!feof($file)) {
  2206. fwrite($newf, fread($file, 1024 * 8), 1024 * 8);
  2207. }
  2208. }
  2209. }else{
  2210. writeLog("error", "organizr could not download $url");
  2211. }
  2212. if ($file) {
  2213. fclose($file);
  2214. writeLog("success", "organizr finished downloading the github zip file");
  2215. }else{
  2216. writeLog("error", "organizr could not download the github zip file");
  2217. }
  2218. if ($newf) {
  2219. fclose($newf);
  2220. writeLog("success", "organizr created upgrade zip file from github zip file");
  2221. }else{
  2222. writeLog("error", "organizr could not create upgrade zip file from github zip file");
  2223. }
  2224. }
  2225. function unzipFile($zipFile){
  2226. $zip = new ZipArchive;
  2227. $extractPath = "upgrade/";
  2228. if($zip->open($extractPath . $zipFile) != "true"){
  2229. writeLog("error", "organizr could not unzip upgrade.zip");
  2230. }else{
  2231. writeLog("success", "organizr unzipped upgrade.zip");
  2232. }
  2233. /* Extract Zip File */
  2234. $zip->extractTo($extractPath);
  2235. $zip->close();
  2236. }
  2237. // Function to remove folders and files
  2238. function rrmdir($dir) {
  2239. if (is_dir($dir)) {
  2240. $files = scandir($dir);
  2241. foreach ($files as $file)
  2242. if ($file != "." && $file != "..") rrmdir("$dir/$file");
  2243. rmdir($dir);
  2244. }
  2245. else if (file_exists($dir)) unlink($dir);
  2246. }
  2247. // Function to Copy folders and files
  2248. function rcopy($src, $dst) {
  2249. if (is_dir ( $src )) {
  2250. if (!file_exists($dst)) : mkdir ( $dst ); endif;
  2251. $files = scandir ( $src );
  2252. foreach ( $files as $file )
  2253. if ($file != "." && $file != "..")
  2254. rcopy ( "$src/$file", "$dst/$file" );
  2255. } else if (file_exists ( $src ))
  2256. copy ( $src, $dst );
  2257. }
  2258. $url = 'https://github.com/causefx/Organizr/archive/'.$branch.'.zip';
  2259. $file = "upgrade.zip";
  2260. $source = __DIR__ . '/upgrade/Organizr-'.$branch.'/';
  2261. $cleanup = __DIR__ . "/upgrade/";
  2262. $destination = __DIR__ . "/";
  2263. writeLog("success", "starting organizr upgrade process");
  2264. downloadFile($url, $file);
  2265. unzipFile($file);
  2266. rcopy($source, $destination);
  2267. writeLog("success", "new organizr files copied");
  2268. rrmdir($cleanup);
  2269. writeLog("success", "organizr upgrade folder removed");
  2270. writeLog("success", "organizr has been updated");
  2271. return true;
  2272. }
  2273. // NzbGET Items
  2274. function nzbgetConnect($list = 'listgroups') {
  2275. $url = qualifyURL(NZBGETURL);
  2276. $api = curl_get($url.'/'.NZBGETUSERNAME.':'.NZBGETPASSWORD.'/jsonrpc/'.$list);
  2277. $api = json_decode($api, true);
  2278. $gotNZB = array();
  2279. if (is_array($api) || is_object($api)){
  2280. foreach ($api['result'] AS $child) {
  2281. $downloadName = htmlentities($child['NZBName'], ENT_QUOTES);
  2282. $downloadStatus = $child['Status'];
  2283. $downloadCategory = $child['Category'];
  2284. if($list == "history"){ $downloadPercent = "100"; $progressBar = ""; }
  2285. if($list == "listgroups"){ $downloadPercent = (($child['FileSizeMB'] - $child['RemainingSizeMB']) / $child['FileSizeMB']) * 100; $progressBar = "progress-bar-striped active"; }
  2286. if($child['Health'] <= "750"){
  2287. $downloadHealth = "danger";
  2288. }elseif($child['Health'] <= "900"){
  2289. $downloadHealth = "warning";
  2290. }elseif($child['Health'] <= "1000"){
  2291. $downloadHealth = "success";
  2292. }
  2293. $gotNZB[] = '<tr>
  2294. <td class="col-xs-7 nzbtable-file-row">'.$downloadName.'</td>
  2295. <td class="col-xs-2 nzbtable nzbtable-row">'.$downloadStatus.'</td>
  2296. <td class="col-xs-1 nzbtable nzbtable-row">'.$downloadCategory.'</td>
  2297. <td class="col-xs-2 nzbtable nzbtable-row">
  2298. <div class="progress">
  2299. <div class="progress-bar progress-bar-'.$downloadHealth.' '.$progressBar.'" role="progressbar" aria-valuenow="'.$downloadPercent.'" aria-valuemin="0" aria-valuemax="100" style="width: '.$downloadPercent.'%">
  2300. <p class="text-center">'.round($downloadPercent).'%</p>
  2301. <span class="sr-only">'.$downloadPercent.'% Complete</span>
  2302. </div>
  2303. </div>
  2304. </td>
  2305. </tr>';
  2306. }
  2307. if ($gotNZB) {
  2308. return implode('',$gotNZB);
  2309. } else {
  2310. return '<tr><td colspan="4"><p class="text-center">No Results</p></td></tr>';
  2311. }
  2312. }else{
  2313. writeLog("error", "NZBGET ERROR: could not connect - check URL and/or check token and/or Username and Password - if HTTPS, is cert valid");
  2314. }
  2315. }
  2316. // Sabnzbd Items
  2317. function sabnzbdConnect($list = 'queue') {
  2318. $url = qualifyURL(SABNZBDURL);
  2319. $api = file_get_contents($url.'/api?mode='.$list.'&output=json&apikey='.SABNZBDKEY);
  2320. $api = json_decode($api, true);
  2321. $gotNZB = array();
  2322. foreach ($api[$list]['slots'] AS $child) {
  2323. if($list == "queue"){ $downloadName = $child['filename']; $downloadCategory = $child['cat']; $downloadPercent = (($child['mb'] - $child['mbleft']) / $child['mb']) * 100; $progressBar = "progress-bar-striped active"; }
  2324. if($list == "history"){ $downloadName = $child['name']; $downloadCategory = $child['category']; $downloadPercent = "100"; $progressBar = ""; }
  2325. $downloadStatus = $child['status'];
  2326. $gotNZB[] = '<tr>
  2327. <td class="col-xs-7 nzbtable-file-row">'.$downloadName.'</td>
  2328. <td class="col-xs-2 nzbtable nzbtable-row">'.$downloadStatus.'</td>
  2329. <td class="col-xs-1 nzbtable nzbtable-row">'.$downloadCategory.'</td>
  2330. <td class="col-xs-2 nzbtable nzbtable-row">
  2331. <div class="progress">
  2332. <div class="progress-bar progress-bar-success '.$progressBar.'" role="progressbar" aria-valuenow="'.$downloadPercent.'" aria-valuemin="0" aria-valuemax="100" style="width: '.$downloadPercent.'%">
  2333. <p class="text-center">'.round($downloadPercent).'%</p>
  2334. <span class="sr-only">'.$downloadPercent.'% Complete</span>
  2335. </div>
  2336. </div>
  2337. </td>
  2338. </tr>';
  2339. }
  2340. if ($gotNZB) {
  2341. return implode('',$gotNZB);
  2342. } else {
  2343. return '<tr><td colspan="4"><p class="text-center">No Results</p></td></tr>';
  2344. }
  2345. }
  2346. // Apply new tab settings
  2347. function updateTabs($tabs) {
  2348. if (!isset($GLOBALS['file_db'])) {
  2349. $GLOBALS['file_db'] = new PDO('sqlite:'.DATABASE_LOCATION.'users.db');
  2350. $GLOBALS['file_db']->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  2351. }
  2352. // Validate
  2353. if (!isset($tabs['defaultz'])) { $tabs['defaultz'][current(array_keys($tabs['name']))] = 'true'; }
  2354. if (isset($tabs['name']) && isset($tabs['url']) && is_array($tabs['name'])) {
  2355. // Clear Existing Tabs
  2356. $GLOBALS['file_db']->query("DELETE FROM tabs");
  2357. // Process New Tabs
  2358. $totalValid = 0;
  2359. foreach ($tabs['name'] as $key => $value) {
  2360. // Qualify
  2361. if (!$value || !isset($tabs['url']) || !$tabs['url'][$key]) { continue; }
  2362. $totalValid++;
  2363. $fields = array();
  2364. foreach(array('id','name','url','icon','iconurl','order','ping_url') as $v) {
  2365. if (isset($tabs[$v]) && isset($tabs[$v][$key])) { $fields[$v] = $tabs[$v][$key]; }
  2366. }
  2367. foreach(array('active','user','guest','defaultz','window','splash','ping') as $v) {
  2368. if (isset($tabs[$v]) && isset($tabs[$v][$key])) { $fields[$v] = ($tabs[$v][$key]!=='false'?'true':'false'); }
  2369. }
  2370. $GLOBALS['file_db']->query('INSERT INTO tabs (`'.implode('`,`',array_keys($fields)).'`) VALUES (\''.implode("','",$fields).'\');');
  2371. }
  2372. writeLog("success", "tabs successfully saved");
  2373. return $totalValid;
  2374. } else {
  2375. writeLog("error", "tabs could not save");
  2376. return false;
  2377. }
  2378. writeLog("error", "tabs could not save");
  2379. return false;
  2380. }
  2381. // ==============
  2382. function clean($strin) {
  2383. $strout = null;
  2384. for ($i = 0; $i < strlen($strin); $i++) {
  2385. $ord = ord($strin[$i]);
  2386. if (($ord > 0 && $ord < 32) || ($ord >= 127)) {
  2387. $strout .= "&amp;#{$ord};";
  2388. }
  2389. else {
  2390. switch ($strin[$i]) {
  2391. case '<':
  2392. $strout .= '&lt;';
  2393. break;
  2394. case '>':
  2395. $strout .= '&gt;';
  2396. break;
  2397. case '&':
  2398. $strout .= '&amp;';
  2399. break;
  2400. case '"':
  2401. $strout .= '&quot;';
  2402. break;
  2403. default:
  2404. $strout .= $strin[$i];
  2405. }
  2406. }
  2407. }
  2408. return $strout;
  2409. }
  2410. function registration_callback($username, $email, $userdir){
  2411. global $data;
  2412. $data = array($username, $email, $userdir);
  2413. }
  2414. function printArray($arrayName){
  2415. $messageCount = count($arrayName);
  2416. $i = 0;
  2417. foreach ( $arrayName as $item ) :
  2418. $i++;
  2419. if($i < $messageCount) :
  2420. echo "<small class='text-uppercase'>" . $item . "</small> & ";
  2421. elseif($i = $messageCount) :
  2422. echo "<small class='text-uppercase'>" . $item . "</small>";
  2423. endif;
  2424. endforeach;
  2425. }
  2426. function write_ini_file($content, $path) {
  2427. if (!$handle = fopen($path, 'w')) {
  2428. return false;
  2429. }
  2430. $success = fwrite($handle, trim($content));
  2431. fclose($handle);
  2432. return $success;
  2433. }
  2434. function gotTimezone(){
  2435. $regions = array(
  2436. 'Africa' => DateTimeZone::AFRICA,
  2437. 'America' => DateTimeZone::AMERICA,
  2438. 'Antarctica' => DateTimeZone::ANTARCTICA,
  2439. 'Arctic' => DateTimeZone::ARCTIC,
  2440. 'Asia' => DateTimeZone::ASIA,
  2441. 'Atlantic' => DateTimeZone::ATLANTIC,
  2442. 'Australia' => DateTimeZone::AUSTRALIA,
  2443. 'Europe' => DateTimeZone::EUROPE,
  2444. 'Indian' => DateTimeZone::INDIAN,
  2445. 'Pacific' => DateTimeZone::PACIFIC
  2446. );
  2447. $timezones = array();
  2448. foreach ($regions as $name => $mask) {
  2449. $zones = DateTimeZone::listIdentifiers($mask);
  2450. foreach($zones as $timezone) {
  2451. $time = new DateTime(NULL, new DateTimeZone($timezone));
  2452. $ampm = $time->format('H') > 12 ? ' ('. $time->format('g:i a'). ')' : '';
  2453. $timezones[$name][$timezone] = substr($timezone, strlen($name) + 1) . ' - ' . $time->format('H:i') . $ampm;
  2454. }
  2455. }
  2456. print '<select name="timezone" id="timezone" class="form-control material input-sm" required>';
  2457. foreach($timezones as $region => $list) {
  2458. print '<optgroup label="' . $region . '">' . "\n";
  2459. foreach($list as $timezone => $name) {
  2460. if($timezone == TIMEZONE) : $selected = " selected"; else : $selected = ""; endif;
  2461. print '<option value="' . $timezone . '"' . $selected . '>' . $name . '</option>' . "\n";
  2462. }
  2463. print '</optgroup>' . "\n";
  2464. }
  2465. print '</select>';
  2466. }
  2467. function getTimezone(){
  2468. $regions = array(
  2469. 'Africa' => DateTimeZone::AFRICA,
  2470. 'America' => DateTimeZone::AMERICA,
  2471. 'Antarctica' => DateTimeZone::ANTARCTICA,
  2472. 'Arctic' => DateTimeZone::ARCTIC,
  2473. 'Asia' => DateTimeZone::ASIA,
  2474. 'Atlantic' => DateTimeZone::ATLANTIC,
  2475. 'Australia' => DateTimeZone::AUSTRALIA,
  2476. 'Europe' => DateTimeZone::EUROPE,
  2477. 'Indian' => DateTimeZone::INDIAN,
  2478. 'Pacific' => DateTimeZone::PACIFIC
  2479. );
  2480. $timezones = array();
  2481. foreach ($regions as $name => $mask) {
  2482. $zones = DateTimeZone::listIdentifiers($mask);
  2483. foreach($zones as $timezone) {
  2484. $time = new DateTime(NULL, new DateTimeZone($timezone));
  2485. $ampm = $time->format('H') > 12 ? ' ('. $time->format('g:i a'). ')' : '';
  2486. $timezones[$name][$timezone] = substr($timezone, strlen($name) + 1) . ' - ' . $time->format('H:i') . $ampm;
  2487. }
  2488. }
  2489. print '<select name="timezone" id="timezone" class="form-control material" required>';
  2490. foreach($timezones as $region => $list) {
  2491. print '<optgroup label="' . $region . '">' . "\n";
  2492. foreach($list as $timezone => $name) {
  2493. print '<option value="' . $timezone . '">' . $name . '</option>' . "\n";
  2494. }
  2495. print '</optgroup>' . "\n";
  2496. }
  2497. print '</select>';
  2498. }
  2499. function explosion($string, $position){
  2500. $getWord = explode("|", $string);
  2501. return $getWord[$position];
  2502. }
  2503. function getServerPath() {
  2504. if (isset($_SERVER['HTTP_X_FORWARDED_PROTO']) && $_SERVER['HTTP_X_FORWARDED_PROTO'] == "https"){
  2505. $protocol = "https://";
  2506. }elseif (isset($_SERVER['HTTPS']) && $_SERVER['HTTPS'] != 'off') {
  2507. $protocol = "https://";
  2508. } else {
  2509. $protocol = "http://";
  2510. }
  2511. $domain = '';
  2512. if (isset($_SERVER['SERVER_NAME']) && strpos($_SERVER['SERVER_NAME'], '.') !== false){
  2513. $domain = $_SERVER['SERVER_NAME'];
  2514. }elseif(isset($_SERVER['HTTP_HOST'])){
  2515. if (strpos($_SERVER['HTTP_HOST'], ':') !== false) {
  2516. $domain = explode(':', $_SERVER['HTTP_HOST'])[0];
  2517. $port = explode(':', $_SERVER['HTTP_HOST'])[1];
  2518. if ($port == "80" || $port == "443"){
  2519. $domain = $domain;
  2520. }else{
  2521. $domain = $_SERVER['HTTP_HOST'];
  2522. }
  2523. }else{
  2524. $domain = $_SERVER['HTTP_HOST'];
  2525. }
  2526. }
  2527. return $protocol . $domain . dirname($_SERVER['REQUEST_URI']);
  2528. }
  2529. function get_browser_name() {
  2530. $user_agent = $_SERVER['HTTP_USER_AGENT'];
  2531. if (strpos($user_agent, 'Opera') || strpos($user_agent, 'OPR/')) return 'Opera';
  2532. elseif (strpos($user_agent, 'Edge')) return 'Edge';
  2533. elseif (strpos($user_agent, 'Chrome')) return 'Chrome';
  2534. elseif (strpos($user_agent, 'Safari')) return 'Safari';
  2535. elseif (strpos($user_agent, 'Firefox')) return 'Firefox';
  2536. elseif (strpos($user_agent, 'MSIE') || strpos($user_agent, 'Trident/7')) return 'Internet Explorer';
  2537. return 'Other';
  2538. }
  2539. function getSickrageCalendarWanted($array){
  2540. $array = json_decode($array, true);
  2541. //$gotCalendar = "";
  2542. $gotCalendar = array();
  2543. $i = 0;
  2544. foreach($array['data']['missed'] AS $child) {
  2545. $i++;
  2546. $seriesName = $child['show_name'];
  2547. $episodeID = $child['tvdbid'];
  2548. $episodeAirDate = $child['airdate'];
  2549. $episodeAirDateTime = explode(" ",$child['airs']);
  2550. $episodeAirDateTime = date("H:i:s", strtotime($episodeAirDateTime[1].$episodeAirDateTime[2]));
  2551. $episodeAirDate = strtotime($episodeAirDate.$episodeAirDateTime);
  2552. $episodeAirDate = date("Y-m-d H:i:s", $episodeAirDate);
  2553. if (new DateTime() < new DateTime($episodeAirDate)) { $unaired = true; }
  2554. $downloaded = "0";
  2555. if($downloaded == "0" && isset($unaired)){ $downloaded = "indigo-bg"; }elseif($downloaded == "1"){ $downloaded = "green-bg";}else{ $downloaded = "red-bg"; }
  2556. //$gotCalendar .= "{ title: \"$seriesName\", start: \"$episodeAirDate\", className: \"$downloaded tvID--$episodeID\", imagetype: \"tv\" }, \n";
  2557. array_push($gotCalendar, array(
  2558. "id" => "Sick-Miss-".$i,
  2559. "title" => $seriesName,
  2560. "start" => $episodeAirDate,
  2561. "className" => $downloaded." tvID--".$episodeID,
  2562. "imagetype" => "tv",
  2563. ));
  2564. }
  2565. foreach($array['data']['today'] AS $child) {
  2566. $i++;
  2567. $seriesName = $child['show_name'];
  2568. $episodeID = $child['tvdbid'];
  2569. $episodeAirDate = $child['airdate'];
  2570. $episodeAirDateTime = explode(" ",$child['airs']);
  2571. $episodeAirDateTime = date("H:i:s", strtotime($episodeAirDateTime[1].$episodeAirDateTime[2]));
  2572. $episodeAirDate = strtotime($episodeAirDate.$episodeAirDateTime);
  2573. $episodeAirDate = date("Y-m-d H:i:s", $episodeAirDate);
  2574. if (new DateTime() < new DateTime($episodeAirDate)) { $unaired = true; }
  2575. $downloaded = "0";
  2576. if($downloaded == "0" && isset($unaired)){ $downloaded = "indigo-bg"; }elseif($downloaded == "1"){ $downloaded = "green-bg";}else{ $downloaded = "red-bg"; }
  2577. //$gotCalendar .= "{ title: \"$seriesName\", start: \"$episodeAirDate\", className: \"$downloaded tvID--$episodeID\", imagetype: \"tv\" }, \n";
  2578. array_push($gotCalendar, array(
  2579. "id" => "Sick-Today-".$i,
  2580. "title" => $seriesName,
  2581. "start" => $episodeAirDate,
  2582. "className" => $downloaded." tvID--".$episodeID,
  2583. "imagetype" => "tv",
  2584. ));
  2585. }
  2586. foreach($array['data']['soon'] AS $child) {
  2587. $i++;
  2588. $seriesName = $child['show_name'];
  2589. $episodeID = $child['tvdbid'];
  2590. $episodeAirDate = $child['airdate'];
  2591. $episodeAirDateTime = explode(" ",$child['airs']);
  2592. $episodeAirDateTime = date("H:i:s", strtotime($episodeAirDateTime[1].$episodeAirDateTime[2]));
  2593. $episodeAirDate = strtotime($episodeAirDate.$episodeAirDateTime);
  2594. $episodeAirDate = date("Y-m-d H:i:s", $episodeAirDate);
  2595. if (new DateTime() < new DateTime($episodeAirDate)) { $unaired = true; }
  2596. $downloaded = "0";
  2597. if($downloaded == "0" && isset($unaired)){ $downloaded = "indigo-bg"; }elseif($downloaded == "1"){ $downloaded = "green-bg";}else{ $downloaded = "red-bg"; }
  2598. //$gotCalendar .= "{ title: \"$seriesName\", start: \"$episodeAirDate\", className: \"$downloaded tvID--$episodeID\", imagetype: \"tv\" }, \n";
  2599. array_push($gotCalendar, array(
  2600. "id" => "Sick-Soon-".$i,
  2601. "title" => $seriesName,
  2602. "start" => $episodeAirDate,
  2603. "className" => $downloaded." tvID--".$episodeID,
  2604. "imagetype" => "tv",
  2605. ));
  2606. }
  2607. foreach($array['data']['later'] AS $child) {
  2608. $i++;
  2609. $seriesName = $child['show_name'];
  2610. $episodeID = $child['tvdbid'];
  2611. $episodeAirDate = $child['airdate'];
  2612. $episodeAirDateTime = explode(" ",$child['airs']);
  2613. $episodeAirDateTime = date("H:i:s", strtotime($episodeAirDateTime[1].$episodeAirDateTime[2]));
  2614. $episodeAirDate = strtotime($episodeAirDate.$episodeAirDateTime);
  2615. $episodeAirDate = date("Y-m-d H:i:s", $episodeAirDate);
  2616. if (new DateTime() < new DateTime($episodeAirDate)) { $unaired = true; }
  2617. $downloaded = "0";
  2618. if($downloaded == "0" && isset($unaired)){ $downloaded = "indigo-bg"; }elseif($downloaded == "1"){ $downloaded = "green-bg";}else{ $downloaded = "red-bg"; }
  2619. //$gotCalendar .= "{ title: \"$seriesName\", start: \"$episodeAirDate\", className: \"$downloaded tvID--$episodeID\", imagetype: \"tv\" }, \n";
  2620. array_push($gotCalendar, array(
  2621. "id" => "Sick-Later-".$i,
  2622. "title" => $seriesName,
  2623. "start" => $episodeAirDate,
  2624. "className" => $downloaded." tvID--".$episodeID,
  2625. "imagetype" => "tv",
  2626. ));
  2627. }
  2628. if ($i != 0){ return $gotCalendar; }
  2629. }
  2630. function getSickrageCalendarHistory($array){
  2631. $array = json_decode($array, true);
  2632. //$gotCalendar = "";
  2633. $gotCalendar = array();
  2634. $i = 0;
  2635. foreach($array['data'] AS $child) {
  2636. $i++;
  2637. $seriesName = $child['show_name'];
  2638. $episodeID = $child['tvdbid'];
  2639. $episodeAirDate = $child['date'];
  2640. $downloaded = "green-bg";
  2641. //$gotCalendar .= "{ title: \"$seriesName\", start: \"$episodeAirDate\", className: \"$downloaded tvID--$episodeID\", imagetype: \"tv\" }, \n";
  2642. array_push($gotCalendar, array(
  2643. "id" => "Sick-History-".$i,
  2644. "title" => $seriesName,
  2645. "start" => $episodeAirDate,
  2646. "className" => $downloaded." tvID--".$episodeID,
  2647. "imagetype" => "tv",
  2648. ));
  2649. }
  2650. if ($i != 0){ return $gotCalendar; }
  2651. }
  2652. function getSonarrCalendar($array){
  2653. $array = json_decode($array, true);
  2654. //$gotCalendar = "";
  2655. $gotCalendar = array();
  2656. $i = 0;
  2657. foreach($array AS $child) {
  2658. $i++;
  2659. $seriesName = $child['series']['title'];
  2660. $episodeID = $child['series']['tvdbId'];
  2661. if(!isset($episodeID)){ $episodeID = ""; }
  2662. $episodeName = htmlentities($child['title'], ENT_QUOTES);
  2663. if($child['episodeNumber'] == "1"){ $episodePremier = "true"; }else{ $episodePremier = "false"; }
  2664. $episodeAirDate = $child['airDateUtc'];
  2665. $episodeAirDate = strtotime($episodeAirDate);
  2666. $episodeAirDate = date("Y-m-d H:i:s", $episodeAirDate);
  2667. if (new DateTime() < new DateTime($episodeAirDate)) { $unaired = true; }
  2668. $downloaded = $child['hasFile'];
  2669. if($downloaded == "0" && isset($unaired) && $episodePremier == "true"){ $downloaded = "light-blue-bg"; }elseif($downloaded == "0" && isset($unaired)){ $downloaded = "indigo-bg"; }elseif($downloaded == "1"){ $downloaded = "green-bg";}else{ $downloaded = "red-bg"; }
  2670. //$gotCalendar .= "{ title: \"$seriesName\", start: \"$episodeAirDate\", className: \"$downloaded tvID--$episodeID\", imagetype: \"tv\" }, \n";
  2671. array_push($gotCalendar, array(
  2672. "id" => "Sonarr-".$i,
  2673. "title" => $seriesName,
  2674. "start" => $episodeAirDate,
  2675. "className" => $downloaded." tvID--".$episodeID,
  2676. "imagetype" => "tv",
  2677. ));
  2678. }
  2679. if ($i != 0){ return $gotCalendar; }
  2680. }
  2681. function getCouchCalendar(){
  2682. $url = qualifyURL(COUCHURL);
  2683. $api = curl_get($url."/api/".COUCHAPI."/media.list");
  2684. $api = json_decode($api, true);
  2685. $i = 0;
  2686. $gotCalendar = array();
  2687. if (is_array($api) || is_object($api)){
  2688. foreach($api['movies'] AS $child) {
  2689. if($child['status'] == "active" || $child['status'] == "done" ){
  2690. $i++;
  2691. $movieName = $child['info']['original_title'];
  2692. $movieID = $child['info']['tmdb_id'];
  2693. if(!isset($movieID)){ $movieID = ""; }
  2694. $physicalRelease = (isset($child['info']['released']) ? $child['info']['released'] : null);
  2695. $backupRelease = (isset($child['info']['release_date']['theater']) ? $child['info']['release_date']['theater'] : null);
  2696. $physicalRelease = (isset($physicalRelease) ? $physicalRelease : $backupRelease);
  2697. $physicalRelease = strtotime($physicalRelease);
  2698. $physicalRelease = date("Y-m-d", $physicalRelease);
  2699. if (new DateTime() < new DateTime($physicalRelease)) { $notReleased = "true"; }else{ $notReleased = "false"; }
  2700. $downloaded = ($child['status'] == "active") ? "0" : "1";
  2701. if($downloaded == "0" && $notReleased == "true"){ $downloaded = "indigo-bg"; }elseif($downloaded == "1"){ $downloaded = "green-bg"; }else{ $downloaded = "red-bg"; }
  2702. array_push($gotCalendar, array(
  2703. "id" => "CouchPotato-".$i,
  2704. "title" => $movieName,
  2705. "start" => $physicalRelease,
  2706. "className" => $downloaded." movieID--".$movieID,
  2707. "imagetype" => "film",
  2708. ));
  2709. }
  2710. }
  2711. if ($i != 0){ return $gotCalendar; }
  2712. }else{
  2713. writeLog("error", "CouchPotato ERROR: could not connect - check URL and/or check API key - if HTTPS, is cert valid");
  2714. }
  2715. }
  2716. function getRadarrCalendar($array){
  2717. $array = json_decode($array, true);
  2718. $gotCalendar = array();
  2719. $i = 0;
  2720. foreach($array AS $child) {
  2721. if(isset($child['physicalRelease'])){
  2722. $i++;
  2723. $movieName = $child['title'];
  2724. $movieID = $child['tmdbId'];
  2725. if(!isset($movieID)){ $movieID = ""; }
  2726. $physicalRelease = $child['physicalRelease'];
  2727. $physicalRelease = strtotime($physicalRelease);
  2728. $physicalRelease = date("Y-m-d", $physicalRelease);
  2729. if (new DateTime() < new DateTime($physicalRelease)) { $notReleased = "true"; }else{ $notReleased = "false"; }
  2730. $downloaded = $child['hasFile'];
  2731. if($downloaded == "0" && $notReleased == "true"){ $downloaded = "indigo-bg"; }elseif($downloaded == "1"){ $downloaded = "green-bg"; }else{ $downloaded = "red-bg"; }
  2732. array_push($gotCalendar, array(
  2733. "id" => "Radarr-".$i,
  2734. "title" => $movieName,
  2735. "start" => $physicalRelease,
  2736. "className" => $downloaded." movieID--".$movieID,
  2737. "imagetype" => "film",
  2738. ));
  2739. }
  2740. }
  2741. if ($i != 0){ return $gotCalendar; }
  2742. }
  2743. function getHeadphonesCalendar($url, $key, $list){
  2744. $url = qualifyURL(HEADPHONESURL);
  2745. $api = curl_get($url."/api?apikey=".$key."&cmd=$list");
  2746. $api = json_decode($api, true);
  2747. $i = 0;
  2748. //$gotCalendar = "";
  2749. $gotCalendar = array();;
  2750. if (is_array($api) || is_object($api)){
  2751. foreach($api AS $child) {
  2752. if($child['Status'] == "Wanted" && $list == "getWanted" && $child['ReleaseDate']){
  2753. $i++;
  2754. $albumName = addslashes($child['AlbumTitle']);
  2755. $albumArtist = htmlentities($child['ArtistName'], ENT_QUOTES);
  2756. $albumDate = (strlen($child['ReleaseDate']) > 4) ? $child['ReleaseDate'] : $child['ReleaseDate']."-01-01";
  2757. $albumID = $child['AlbumID'];
  2758. $albumDate = strtotime($albumDate);
  2759. $albumDate = date("Y-m-d", $albumDate);
  2760. $albumStatus = $child['Status'];
  2761. if (new DateTime() < new DateTime($albumDate)) { $notReleased = "true"; }else{ $notReleased = "false"; }
  2762. if($albumStatus == "Wanted" && $notReleased == "true"){ $albumStatusColor = "indigo-bg"; }elseif($albumStatus == "Downloaded"){ $albumStatusColor = "green-bg"; }else{ $albumStatusColor = "red-bg"; }
  2763. //$gotCalendar .= "{ title: \"$albumArtist - $albumName\", start: \"$albumDate\", className: \"$albumStatusColor\", imagetype: \"music\", url: \"https://musicbrainz.org/release-group/$albumID\" }, \n";
  2764. array_push($gotCalendar, array(
  2765. "id" => "Headphones-".$i,
  2766. "title" => $albumArtist.' - '.$albumName,
  2767. "start" => $albumDate,
  2768. "className" => $albumStatusColor,
  2769. "imagetype" => "music",
  2770. 'url' => "https://musicbrainz.org/release-group/".$albumID,
  2771. ));
  2772. }
  2773. if($child['Status'] == "Processed" && $list == "getHistory"){
  2774. $i++;
  2775. $find = array('_','[', ']', '\n');
  2776. $replace = array(' ','(', ')', ' ');
  2777. $albumName = addslashes(str_replace($find,$replace,$child['FolderName']));
  2778. $albumDate = $child['DateAdded'];
  2779. $albumID = $child['AlbumID'];
  2780. $albumDate = strtotime($albumDate);
  2781. $albumDate = date("Y-m-d", $albumDate);
  2782. $albumStatusColor = "green-bg";
  2783. if (new DateTime() < new DateTime($albumDate)) { $notReleased = "true"; }else{ $notReleased = "false"; }
  2784. //$gotCalendar .= "{ title: \"$albumName\", start: \"$albumDate\", className: \"$albumStatusColor\", imagetype: \"music\", url: \"https://musicbrainz.org/release-group/$albumID\" }, \n";
  2785. array_push($gotCalendar, array(
  2786. "id" => "Headphones-".$i,
  2787. "title" => $albumName,
  2788. "start" => $albumDate,
  2789. "className" => $albumStatusColor,
  2790. "imagetype" => "music",
  2791. 'url' => "https://musicbrainz.org/release-group/".$albumID,
  2792. ));
  2793. }
  2794. }
  2795. if ($i != 0){ return $gotCalendar; }
  2796. }else{
  2797. writeLog("error", "HEADPHONES $list ERROR: could not connect - check URL and/or check API key - if HTTPS, is cert valid");
  2798. }
  2799. }
  2800. function checkRootPath($string){
  2801. if($string == "\\" || $string == "/"){
  2802. return "/";
  2803. }else{
  2804. return str_replace("\\", "/", $string) . "/";
  2805. }
  2806. }
  2807. function strip($string){
  2808. $string = strip_tags($string);
  2809. return preg_replace('/[ \t]+/', ' ', preg_replace('/\s*$^\s*/m', "\n", $string));
  2810. }
  2811. function writeLog($type, $message){
  2812. if(file_exists("org.log")){
  2813. if(filesize("org.log") > 500000){
  2814. rename('org.log','org['.date('Y-m-d').'].log');
  2815. $message2 = date("Y-m-d H:i:s")."|".$type."|".strip("ORG LOG: Creating backup of org.log to org[".date('Y-m-d')."].log ")."\n";
  2816. file_put_contents("org.log", $message2, FILE_APPEND | LOCK_EX);
  2817. }
  2818. }
  2819. $message = date("Y-m-d H:i:s")."|".$type."|".strip($message)."\n";
  2820. file_put_contents("org.log", $message, FILE_APPEND | LOCK_EX);
  2821. }
  2822. function readLog(){
  2823. $log = file("org.log",FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES);
  2824. $log = array_reverse($log);
  2825. foreach($log as $line){
  2826. if(substr_count($line, '|') == 2){
  2827. $line = explode("|", strip($line));
  2828. $line[1] = ($line[1] == "error") ? '<span class="label label-danger">Error</span>' : '<span class="label label-primary">Success</span>';
  2829. echo "<tr><td>".$line[0]."</td><td>".$line[2]."</td><td>".$line[1]."</td></tr>";
  2830. }
  2831. }
  2832. }
  2833. function buildStream($array){
  2834. $result = "";
  2835. if (array_key_exists('platform', $array)) {
  2836. $result .= '<div class="reg-info" style="margin-top:0; padding-left:0; position: absolute; bottom: 10px; left: 10px;"><div style="margin-right: 0;" class="item pull-left text-center"><img alt="'.$array['platform'].'" class="img-circle" height="55px" src="images/platforms/'.getPlatform($array['platform']).'"></div></div><div class="clearfix"></div>';
  2837. }
  2838. if (array_key_exists('device', $array)) {
  2839. $result .= '<div class="reg-info" style="margin-top:0; padding-left:5%;"><div style="margin-right: 0;" class="item pull-left text-center"><span style="font-size: 15px;" class="block text-center"><i class="fa fa-laptop fa-fw"></i>'.$array['device'].'</span></div></div><div class="clearfix"></div>';
  2840. }
  2841. if (array_key_exists('stream', $array)) {
  2842. $result .= '<div class="reg-info" style="margin-top:0; padding-left:5%;"><div style="margin-right: 0;" class="item pull-left text-center"><span style="font-size: 15px;" class="block text-center"><i class="fa fa-play fa-fw"></i>'.$array['stream'].'</span></div></div><div class="clearfix"></div>';
  2843. }
  2844. if (array_key_exists('video', $array)) {
  2845. $result .= '<div class="reg-info" style="margin-top:0; padding-left:5%;"><div style="margin-right: 0;" class="item pull-left text-center"><span style="font-size: 15px;" class="block text-center"><i class="fa fa-film fa-fw"></i>'.$array['video'].'</span></div></div><div class="clearfix"></div>';
  2846. }
  2847. if (array_key_exists('audio', $array)) {
  2848. $result .= '<div class="reg-info" style="margin-top:0; padding-left:5%;"><div style="margin-right: 0;" class="item pull-left text-center"><span style="font-size: 15px;" class="block text-center"><i class="fa fa-volume-up fa-fw"></i>'.$array['audio'].'</span></div></div><div class="clearfix"></div>';
  2849. }
  2850. return $result;
  2851. }
  2852. function streamType($value){
  2853. if($value == "transcode" || $value == "Transcode"){
  2854. return "Transcode";
  2855. }elseif($value == "copy" || $value == "DirectStream"){
  2856. return "Direct Stream";
  2857. }elseif($value == "directplay" || $value == "DirectPlay"){
  2858. return "Direct Play";
  2859. }else{
  2860. return "Direct Play";
  2861. }
  2862. }
  2863. function getPlatform($platform){
  2864. $allPlatforms = array(
  2865. "Chrome" => "chrome.png",
  2866. "tvOS" => "atv.png",
  2867. "iOS" => "ios.png",
  2868. "Xbox One" => "xbox.png",
  2869. "Mystery 4" => "playstation.png",
  2870. "Samsung" => "samsung.png",
  2871. "Roku" => "roku.png",
  2872. "Emby for iOS" => "ios.png",
  2873. "Emby Mobile" => "emby.png",
  2874. "Emby Theater" => "emby.png",
  2875. "Emby Classic" => "emby.png",
  2876. "Safari" => "safari.png",
  2877. "Android" => "android.png",
  2878. "AndroidTv" => "android.png",
  2879. "Chromecast" => "chromecast.png",
  2880. "Dashboard" => "emby.png",
  2881. "Dlna" => "dlna.png",
  2882. "Windows Phone" => "wp.png",
  2883. "Windows RT" => "win8.png",
  2884. "Kodi" => "kodi.png",
  2885. );
  2886. if (array_key_exists($platform, $allPlatforms)) {
  2887. return $allPlatforms[$platform];
  2888. }else{
  2889. return "pmp.png";
  2890. }
  2891. }
  2892. function getServer(){
  2893. $server = isset($_SERVER["HTTP_HOST"]) ? $_SERVER["HTTP_HOST"] : $_SERVER["SERVER_NAME"];
  2894. return $server;
  2895. }
  2896. function prettyPrint($array) {
  2897. echo "<pre>";
  2898. print_r($array);
  2899. echo "</pre>";
  2900. echo "<br/>";
  2901. }
  2902. function checkFrame($array, $url){
  2903. if(array_key_exists("x-frame-options", $array)){
  2904. if($array['x-frame-options'] == "deny"){
  2905. return false;
  2906. }elseif($array['x-frame-options'] == "sameorgin"){
  2907. $digest = parse_url($url);
  2908. $host = (isset($digest['host'])?$digest['host']:'');
  2909. if(getServer() == $host){
  2910. return true;
  2911. }else{
  2912. return false;
  2913. }
  2914. }
  2915. }else{
  2916. if(!$array){
  2917. return false;
  2918. }
  2919. return true;
  2920. }
  2921. }
  2922. function frameTest($url){
  2923. $array = array_change_key_case(get_headers(qualifyURL($url), 1));
  2924. $url = qualifyURL($url);
  2925. if(checkFrame($array, $url)){
  2926. return true;
  2927. }else{
  2928. return false;
  2929. }
  2930. }
  2931. function sendResult($result, $icon = "floppy-o", $message = false, $success = "WAS_SUCCESSFUL", $fail = "HAS_FAILED", $send = true) {
  2932. $notifyExplode = explode("-", NOTIFYEFFECT);
  2933. if ($result) {
  2934. $msg = array(
  2935. 'html' => ($message?''.$message.' <strong>'.translate($success).'</strong>':'<strong>'.translate($success).'</strong>'),
  2936. 'icon' => $icon,
  2937. 'type' => 'success',
  2938. 'length' => '5000',
  2939. 'layout' => $notifyExplode[0],
  2940. 'effect' => $notifyExplode[1],
  2941. );
  2942. } else {
  2943. $msg = array(
  2944. 'html' => ($message?''.$message.' <strong>'.translate($fail).'</strong>':'<strong>'.translate($fail).'</strong>'),
  2945. 'icon' => $icon,
  2946. 'type' => 'error',
  2947. 'length' => '5000',
  2948. 'layout' => $notifyExplode[0],
  2949. 'effect' => $notifyExplode[1],
  2950. );
  2951. }
  2952. // Send and kill script?
  2953. if ($send) {
  2954. header('Content-Type: application/json');
  2955. echo json_encode(array('notify'=>$msg));
  2956. die();
  2957. }
  2958. return $msg;
  2959. }
  2960. function buildHomepageNotice($layout, $type, $title, $message){
  2961. switch ($layout) {
  2962. case 'elegant':
  2963. return '
  2964. <div id="homepageNotice" class="row">
  2965. <div class="col-lg-12">
  2966. <div class="content-box big-box box-shadow panel-box panel-'.$type.'">
  2967. <div class="content-title i-block">
  2968. <h4 class="zero-m"><strong>'.$title.'</strong></h4>
  2969. <div class="content-tools i-block pull-right">
  2970. <a class="close-btn">
  2971. <i class="fa fa-times"></i>
  2972. </a>
  2973. </div>
  2974. </div>
  2975. '.$message.'
  2976. </div>
  2977. </div>
  2978. </div>
  2979. ';
  2980. break;
  2981. case 'basic':
  2982. return '
  2983. <div id="homepageNotice" class="row">
  2984. <div class="col-lg-12">
  2985. <div class="panel panel-'.$type.'">
  2986. <div class="panel-heading">
  2987. <h3 class="panel-title">'.$title.'</h3>
  2988. </div>
  2989. <div class="panel-body">
  2990. '.$message.'
  2991. </div>
  2992. </div>
  2993. </div>
  2994. </div>
  2995. ';
  2996. break;
  2997. case 'jumbotron';
  2998. return '
  2999. <div id="homepageNotice" class="row">
  3000. <div class="col-lg-12">
  3001. <div class="jumbotron">
  3002. <div class="container">
  3003. <h1>'.$title.'</h1>
  3004. <p>'.$message.'</p>
  3005. </div>
  3006. </div>
  3007. </div>
  3008. </div>
  3009. ';
  3010. }
  3011. }
  3012. function embyArray($array, $type) {
  3013. $key = ($type == "video" ? "Height" : "Channels");
  3014. if (array_key_exists($key, $array)) {
  3015. switch ($type) {
  3016. case "video":
  3017. $codec = $array["Codec"];
  3018. $height = $array["Height"];
  3019. $width = $array["Width"];
  3020. break;
  3021. default:
  3022. $codec = $array["Codec"];
  3023. $channels = $array["Channels"];
  3024. }
  3025. return ($type == "video" ? "(".$codec.") (".$width."x".$height.")" : "(".$codec.") (".$channels."ch)");
  3026. }
  3027. foreach ($array as $element) {
  3028. if (is_array($element)) {
  3029. if (embyArray($element, $type)) {
  3030. return embyArray($element, $type);
  3031. }
  3032. }
  3033. }
  3034. }
  3035. // Get Now Playing Streams From Plex
  3036. function searchPlex($query){
  3037. $address = qualifyURL(PLEXURL);
  3038. $openTab = (PLEXTABNAME) ? "true" : "false";
  3039. // Perform API requests
  3040. $api = @curl_get($address."/search?query=".rawurlencode($query)."&X-Plex-Token=".PLEXTOKEN);
  3041. libxml_use_internal_errors(true);
  3042. $api = simplexml_load_string($api);
  3043. $getServer = simplexml_load_string(@curl_get($address."/?X-Plex-Token=".PLEXTOKEN));
  3044. if (!$getServer) { return 'Could not load!'; }
  3045. // Identify the local machine
  3046. $server = $getServer['machineIdentifier'];
  3047. $pre = "<table class=\"table table-hover table-stripped\"><thead><tr><th>Cover</th><th>Title</th><th>Genre</th><th>Year</th><th>Type</th><th>Added</th><th>Extra Info</th></tr></thead><tbody>";
  3048. $items = "";
  3049. $albums = $movies = $shows = 0;
  3050. $style = 'style="vertical-align: middle"';
  3051. foreach($api AS $child) {
  3052. if($child['type'] != "artist" && $child['type'] != "episode" && isset($child['librarySectionID'])){
  3053. $time = (string)$child['addedAt'];
  3054. $time = new DateTime("@$time");
  3055. $results = array(
  3056. "title" => (string)$child['title'],
  3057. "image" => (string)$child['thumb'],
  3058. "type" => (string)ucwords($child['type']),
  3059. "year" => (string)$child['year'],
  3060. "key" => (string)$child['ratingKey']."-search",
  3061. "ratingkey" => (string)$child['ratingKey'],
  3062. "genre" => (string)$child->Genre['tag'],
  3063. "added" => $time->format('Y-m-d'),
  3064. "extra" => "",
  3065. );
  3066. switch ($child['type']){
  3067. case "album":
  3068. $push = array(
  3069. "title" => (string)$child['parentTitle']." - ".(string)$child['title'],
  3070. );
  3071. $results = array_replace($results,$push);
  3072. $albums++;
  3073. break;
  3074. case "movie":
  3075. $push = array(
  3076. "extra" => "Content Rating: ".(string)$child['contentRating']."<br/>Movie Rating: ".(string)$child['rating'],
  3077. );
  3078. $results = array_replace($results,$push);
  3079. $movies++;
  3080. break;
  3081. case "show":
  3082. $push = array(
  3083. "extra" => "Seasons: ".(string)$child['childCount']."<br/>Episodes: ".(string)$child['leafCount'],
  3084. );
  3085. $results = array_replace($results,$push);
  3086. $shows++;
  3087. break;
  3088. }
  3089. if (file_exists('images/cache/'.$results['key'].'.jpg')){ $image_url = 'images/cache/'.$results['key'].'.jpg'; }
  3090. if (file_exists('images/cache/'.$results['key'].'.jpg') && (time() - 604800) > filemtime('images/cache/'.$results['key'].'.jpg') || !file_exists('images/cache/'.$results['key'].'.jpg')) {
  3091. $image_url = 'ajax.php?a=plex-image&img='.$results['image'].'&height=150&width=100&key='.$results['key'];
  3092. }
  3093. if(!$results['image']){ $image_url = "images/no-search.png"; $key = "no-search"; }
  3094. if (substr_count(PLEXURL, '.') != 2) {
  3095. $link = "https://app.plex.tv/web/app#!/server/$server/details?key=/library/metadata/".$results['ratingkey'];
  3096. }else{
  3097. $link = PLEXURL."/web/index.html#!/server/$server/details?key=/library/metadata/".$results['ratingkey'];
  3098. }
  3099. $items .= '<tr style="cursor: pointer;" class="openTab" extraTitle="'.$results['title'].'" extraType="'.$child['type'].'" openTab="'.$openTab.'" href="'.$link.'">
  3100. <th scope="row"><img src="'.$image_url.'"></th>
  3101. <td class="col-xs-2 nzbtable nzbtable-row"'.$style.'>'.$results['title'].'</td>
  3102. <td class="col-xs-3 nzbtable nzbtable-row"'.$style.'>'.$results['genre'].'</td>
  3103. <td class="col-xs-1 nzbtable nzbtable-row"'.$style.'>'.$results['year'].'</td>
  3104. <td class="col-xs-1 nzbtable nzbtable-row"'.$style.'>'.$results['type'].'</td>
  3105. <td class="col-xs-3 nzbtable nzbtable-row"'.$style.'>'.$results['added'].'</td>
  3106. <td class="col-xs-2 nzbtable nzbtable-row"'.$style.'>'.$results['extra'].'</td>
  3107. </tr>';
  3108. }
  3109. }
  3110. $totals = '<div style="margin: 10px;" class="sort-todo pull-right">
  3111. <span class="badge gray-bg"><i class="fa fa-film fa-2x white"></i><strong style="
  3112. font-size: 23px;
  3113. ">&nbsp;'.$movies.'</strong></span>
  3114. <span class="badge gray-bg"><i class="fa fa-tv fa-2x white"></i><strong style="
  3115. font-size: 23px;
  3116. ">&nbsp;'.$shows.'</strong></span>
  3117. <span class="badge gray-bg"><i class="fa fa-music fa-2x white"></i><strong style="
  3118. font-size: 23px;
  3119. ">&nbsp;'.$albums.'</strong></span>
  3120. </div>';
  3121. return (!empty($items) ? $totals.$pre.$items."</div></table>" : "<h2 class='text-center'>No Results for $query</h2>" );
  3122. }
  3123. function getBannedUsers($string){
  3124. if (strpos($string, ',') !== false) {
  3125. $banned = explode(",", $string);
  3126. }else{
  3127. $banned = array($string);
  3128. }
  3129. return $banned;
  3130. }
  3131. function getWhitelist($string){
  3132. if (strpos($string, ',') !== false) {
  3133. $whitelist = explode(",", $string);
  3134. }else{
  3135. $whitelist = array($string);
  3136. }
  3137. foreach($whitelist as &$ip){
  3138. $ip = is_numeric(substr($ip, 0, 1)) ? $ip : gethostbyname($ip);
  3139. }
  3140. return $whitelist;
  3141. }
  3142. function get_client_ip() {
  3143. $ipaddress = '';
  3144. if (isset($_SERVER['HTTP_CLIENT_IP']))
  3145. $ipaddress = $_SERVER['HTTP_CLIENT_IP'];
  3146. else if(isset($_SERVER['HTTP_X_FORWARDED_FOR']))
  3147. $ipaddress = $_SERVER['HTTP_X_FORWARDED_FOR'];
  3148. else if(isset($_SERVER['HTTP_X_FORWARDED']))
  3149. $ipaddress = $_SERVER['HTTP_X_FORWARDED'];
  3150. else if(isset($_SERVER['HTTP_FORWARDED_FOR']))
  3151. $ipaddress = $_SERVER['HTTP_FORWARDED_FOR'];
  3152. else if(isset($_SERVER['HTTP_FORWARDED']))
  3153. $ipaddress = $_SERVER['HTTP_FORWARDED'];
  3154. else if(isset($_SERVER['REMOTE_ADDR']))
  3155. $ipaddress = $_SERVER['REMOTE_ADDR'];
  3156. else
  3157. $ipaddress = 'UNKNOWN';
  3158. return $ipaddress;
  3159. }
  3160. //EMAIL SHIT
  3161. function sendEmail($email, $username = "Organizr User", $subject, $body, $cc = null, $bcc = null){
  3162. $mail = new PHPMailer;
  3163. $mail->isSMTP();
  3164. $mail->Host = SMTPHOST;
  3165. $mail->SMTPAuth = SMTPHOSTAUTH;
  3166. $mail->Username = SMTPHOSTUSERNAME;
  3167. $mail->Password = SMTPHOSTPASSWORD;
  3168. $mail->SMTPSecure = SMTPHOSTTYPE;
  3169. $mail->Port = SMTPHOSTPORT;
  3170. $mail->setFrom(SMTPHOSTSENDEREMAIL, SMTPHOSTSENDERNAME);
  3171. $mail->addReplyTo(SMTPHOSTSENDEREMAIL, SMTPHOSTSENDERNAME);
  3172. $mail->isHTML(true);
  3173. if($email){
  3174. $mail->addAddress($email, $username);
  3175. }
  3176. if($cc){
  3177. $mail->addCC($cc);
  3178. }
  3179. if($bcc){
  3180. if(strpos($bcc , ',') === false){
  3181. $mail->addBCC($bcc);
  3182. }else{
  3183. $allEmails = explode(",",$bcc);
  3184. foreach($allEmails as $gotEmail){
  3185. $mail->addBCC($gotEmail);
  3186. }
  3187. }
  3188. }
  3189. $mail->Subject = $subject;
  3190. $mail->Body = $body;
  3191. //$mail->send();
  3192. if(!$mail->send()) {
  3193. writeLog("error", "mail failed to send");
  3194. } else {
  3195. writeLog("success", "mail has been sent");
  3196. }
  3197. }
  3198. //EMAIL SHIT
  3199. function sendTestEmail($to, $from, $host, $auth, $username, $password, $type, $port, $sendername){
  3200. $mail = new PHPMailer;
  3201. $mail->isSMTP();
  3202. $mail->Host = $host;
  3203. $mail->SMTPAuth = $auth;
  3204. $mail->Username = $username;
  3205. $mail->Password = $password;
  3206. $mail->SMTPSecure = $type;
  3207. $mail->Port = $port;
  3208. $mail->setFrom($from, $sendername);
  3209. $mail->addReplyTo($from, $sendername);
  3210. $mail->isHTML(true);
  3211. $mail->addAddress($to, "Organizr Admin");
  3212. $mail->Subject = "Organizr Test E-Mail";
  3213. $mail->Body = "This was just a test!";
  3214. //$mail->send();
  3215. if(!$mail->send()) {
  3216. writeLog("error", "EMAIL TEST: mail failed to send - Error:".$mail->ErrorInfo);
  3217. return false;
  3218. } else {
  3219. writeLog("success", "EMAIL TEST: mail has been sent successfully");
  3220. return true;
  3221. }
  3222. }
  3223. function libraryList(){
  3224. $address = qualifyURL(PLEXURL);
  3225. $headers = array(
  3226. "Accept" => "application/json",
  3227. "X-Plex-Token" => PLEXTOKEN
  3228. );
  3229. libxml_use_internal_errors(true);
  3230. $getServer = simplexml_load_string(@curl_get($address."/?X-Plex-Token=".PLEXTOKEN));
  3231. if (!$getServer) { return 'Could not load!'; }else { $gotServer = $getServer['machineIdentifier']; }
  3232. $api = simplexml_load_string(@curl_get("https://plex.tv/api/servers/$gotServer/shared_servers", $headers));
  3233. $libraryList = array();
  3234. foreach($api->SharedServer->Section AS $child) {
  3235. $libraryList['libraries'][(string)$child['title']] = (string)$child['id'];
  3236. }
  3237. foreach($api->SharedServer AS $child) {
  3238. if(!empty($child['username'])){
  3239. $username = (string)strtolower($child['username']);
  3240. $email = (string)strtolower($child['email']);
  3241. $libraryList['users'][$username] = (string)$child['id'];
  3242. $libraryList['emails'][$email] = (string)$child['id'];
  3243. $libraryList['both'][$username] = $email;
  3244. }
  3245. }
  3246. return (!empty($libraryList) ? array_change_key_case($libraryList,CASE_LOWER) : null );
  3247. }
  3248. function plexUserShare($username){
  3249. $address = qualifyURL(PLEXURL);
  3250. $headers = array(
  3251. "Accept" => "application/json",
  3252. "Content-Type" => "application/json",
  3253. "X-Plex-Token" => PLEXTOKEN
  3254. );
  3255. libxml_use_internal_errors(true);
  3256. $getServer = simplexml_load_string(@curl_get($address."/?X-Plex-Token=".PLEXTOKEN));
  3257. if (!$getServer) { return 'Could not load!'; }else { $gotServer = $getServer['machineIdentifier']; }
  3258. $json = array(
  3259. "server_id" => $gotServer,
  3260. "shared_server" => array(
  3261. //"library_section_ids" => "[26527637]",
  3262. "invited_email" => $username
  3263. )
  3264. );
  3265. $api = curl_post("https://plex.tv/api/servers/$gotServer/shared_servers/", $json, $headers);
  3266. switch ($api['http_code']['http_code']){
  3267. case 400:
  3268. writeLog("error", "PLEX INVITE: $username already has access to the shared libraries");
  3269. $result = "$username already has access to the shared libraries";
  3270. break;
  3271. case 401:
  3272. writeLog("error", "PLEX INVITE: Invalid Plex Token");
  3273. $result = "Invalid Plex Token";
  3274. break;
  3275. case 200:
  3276. writeLog("success", "PLEX INVITE: $username now has access to your Plex Library");
  3277. $result = "$username now has access to your Plex Library";
  3278. break;
  3279. default:
  3280. writeLog("error", "PLEX INVITE: unknown error");
  3281. $result = false;
  3282. }
  3283. return (!empty($result) ? $result : null );
  3284. }
  3285. function plexUserDelete($username){
  3286. $address = qualifyURL(PLEXURL);
  3287. $headers = array(
  3288. "Accept" => "application/json",
  3289. "Content-Type" => "application/json",
  3290. "X-Plex-Token" => PLEXTOKEN
  3291. );
  3292. libxml_use_internal_errors(true);
  3293. $getServer = simplexml_load_string(@curl_get($address."/?X-Plex-Token=".PLEXTOKEN));
  3294. if (!$getServer) { return 'Could not load!'; }else { $gotServer = $getServer['machineIdentifier']; }
  3295. $id = (is_numeric($username) ? $id : convertPlexName($username, "id"));
  3296. $api = curl_delete("https://plex.tv/api/servers/$gotServer/shared_servers/$id", $headers);
  3297. switch ($api['http_code']['http_code']){
  3298. case 401:
  3299. writeLog("error", "PLEX INVITE: Invalid Plex Token");
  3300. $result = "Invalid Plex Token";
  3301. break;
  3302. case 200:
  3303. writeLog("success", "PLEX INVITE: $username doesn't have access to your Plex Library anymore");
  3304. $result = "$username doesn't have access to your Plex Library anymore";
  3305. break;
  3306. default:
  3307. writeLog("error", "PLEX INVITE: unknown error");
  3308. $result = false;
  3309. }
  3310. return (!empty($result) ? $result : null );
  3311. }
  3312. function convertPlexName($user, $type){
  3313. $array = libraryList();
  3314. switch ($type){
  3315. case "username":
  3316. $plexUser = array_search ($user, $array['users']);
  3317. break;
  3318. case "id":
  3319. if (array_key_exists(strtolower($user), $array['users'])) {
  3320. $plexUser = $array['users'][strtolower($user)];
  3321. }
  3322. break;
  3323. default:
  3324. $plexUser = false;
  3325. }
  3326. return (!empty($plexUser) ? $plexUser : null );
  3327. }
  3328. function randomCode($length = 5, $type = null) {
  3329. switch ($type){
  3330. case "alpha":
  3331. $legend = array_merge(range('A', 'Z'));
  3332. break;
  3333. case "numeric":
  3334. $legend = array_merge(range(0,9));
  3335. break;
  3336. default:
  3337. $legend = array_merge(range(0,9),range('A', 'Z'));
  3338. }
  3339. $code = "";
  3340. for($i=0; $i < $length; $i++) {
  3341. $code .= $legend[mt_rand(0, count($legend) - 1)];
  3342. }
  3343. return $code;
  3344. }
  3345. function inviteCodes($action, $code = null, $usedBy = null) {
  3346. if (!isset($GLOBALS['file_db'])) {
  3347. $GLOBALS['file_db'] = new PDO('sqlite:'.DATABASE_LOCATION.'users.db');
  3348. $GLOBALS['file_db']->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  3349. }
  3350. $now = date("Y-m-d H:i:s");
  3351. switch ($action) {
  3352. case "get":
  3353. // Start Array
  3354. $result = array();
  3355. // Database Lookup
  3356. $invites = $GLOBALS['file_db']->query('SELECT * FROM invites WHERE valid = "Yes"');
  3357. // Get Codes
  3358. foreach($invites as $row) {
  3359. array_push($result, $row['code']);
  3360. }
  3361. // Return the Results
  3362. return (!empty($result) ? $result : false );
  3363. break;
  3364. case "check":
  3365. // Start Array
  3366. $result = array();
  3367. // Database Lookup
  3368. $invites = $GLOBALS['file_db']->query('SELECT * FROM invites WHERE valid = "Yes" AND code = "'.$code.'"');
  3369. // Get Codes
  3370. foreach($invites as $row) {
  3371. $result = $row['code'];
  3372. }
  3373. // Return the Results
  3374. return (!empty($result) ? $result : false );
  3375. break;
  3376. case "use":
  3377. $currentIP = get_client_ip();
  3378. $invites = $GLOBALS['file_db']->query('UPDATE invites SET valid = "No", usedby = "'.$usedBy.'", dateused = "'.$now.'", ip = "'.$currentIP.'" WHERE code = "'.$code.'"');
  3379. if(ENABLEMAIL){
  3380. if (!isset($GLOBALS['USER'])) {
  3381. require_once("user.php");
  3382. $GLOBALS['USER'] = new User('registration_callback');
  3383. }
  3384. sendEmail($GLOBALS['USER']->adminEmail, "Admin", "Plex Invite Used", orgEmail("PLEX Invite Used", "Look who joined the cool club", "Admin", "Hey, The User: $usedBy has redeemd their invite code: [$code], their IP Address was: $currentIP", null, null, "What Next?", "Well, That is up to you. You can go check on them if you like."));
  3385. }
  3386. return (!empty($invites) ? true : false );
  3387. break;
  3388. }
  3389. }
  3390. function plexJoin($username, $email, $password){
  3391. $connectURL = 'https://plex.tv/users.json';
  3392. $headers = array(
  3393. 'Accept'=> 'application/json',
  3394. 'Content-Type' => 'application/x-www-form-urlencoded',
  3395. 'X-Plex-Product' => 'Organizr',
  3396. 'X-Plex-Version' => '1.0',
  3397. 'X-Plex-Client-Identifier' => '01010101-10101010',
  3398. );
  3399. $body = array(
  3400. 'user[email]' => $email,
  3401. 'user[username]' => $username,
  3402. 'user[password]' => $password,
  3403. );
  3404. $api = curl_post($connectURL, $body, $headers);
  3405. $json = json_decode($api['content'], true);
  3406. $errors = (!empty($json['errors']) ? true : false);
  3407. $success = (!empty($json['user']) ? true : false);
  3408. //Use This for later
  3409. $usernameError = (!empty($json['errors']['username']) ? $json['errors']['username'][0] : false);
  3410. $emailError = (!empty($json['errors']['email']) ? $json['errors']['email'][0] : false);
  3411. $passwordError = (!empty($json['errors']['password']) ? $json['errors']['password'][0] : false);
  3412. $errorMessage = "";
  3413. if($errors){
  3414. if($usernameError){ $errorMessage .= "[Username Error: ". $usernameError ."]"; }
  3415. if($emailError){ $errorMessage .= "[Email Error: ". $emailError ."]"; }
  3416. if($passwordError){ $errorMessage .= "[Password Error: ". $passwordError ."]"; }
  3417. }
  3418. switch ($api['http_code']['http_code']){
  3419. case 400:
  3420. writeLog("error", "PLEX JOIN: Error: ".$api['http_code']['http_code']." $username already has access to the shared libraries $errorMessage");
  3421. break;
  3422. case 401:
  3423. writeLog("error", "PLEX JOIN: Error: ".$api['http_code']['http_code']." invalid Plex Token $errorMessage");
  3424. break;
  3425. case 422:
  3426. writeLog("error", "PLEX JOIN: Error: ".$api['http_code']['http_code']." user info error $errorMessage");
  3427. break;
  3428. case 429:
  3429. writeLog("error", "PLEX JOIN: Error: ".$api['http_code']['http_code']." too many requests to plex.tv please try later $errorMessage");
  3430. break;
  3431. case 200:
  3432. case 201:
  3433. writeLog("success", "PLEX JOIN: $username now has access to your Plex Library");
  3434. break;
  3435. default:
  3436. writeLog("error", "PLEX JOIN: unknown error, $errorMessage Error: ".$api['http_code']['http_code']);
  3437. }
  3438. //prettyPrint($api);
  3439. //prettyPrint(json_decode($api['content'], true));
  3440. return (!empty($success) && empty($errors) ? true : false );
  3441. }
  3442. function getCert(){
  3443. $url = "http://curl.haxx.se/ca/cacert.pem";
  3444. $file = getcwd()."/config/cacert.pem";
  3445. $directory = getcwd()."/config/";
  3446. @mkdir($directory, 0770, true);
  3447. if(!file_exists($file)){
  3448. file_put_contents( $file, fopen($url, 'r'));
  3449. writeLog("success", "CERT PEM: pem file created");
  3450. }elseif (file_exists($file) && time() - 2592000 > filemtime($file)) {
  3451. file_put_contents( $file, fopen($url, 'r'));
  3452. writeLog("success", "CERT PEM: downloaded new pem file");
  3453. }
  3454. return $file;
  3455. }
  3456. function customCSS(){
  3457. if(CUSTOMCSS == "true") {
  3458. $template_file = "custom.css";
  3459. $file_handle = fopen($template_file, "rb");
  3460. echo "\n";
  3461. echo fread($file_handle, filesize($template_file));
  3462. fclose($file_handle);
  3463. echo "\n";
  3464. }
  3465. }
  3466. function tvdbToken(){
  3467. $headers = array(
  3468. "Accept" => "application/json",
  3469. "Content-Type" => "application/json"
  3470. );
  3471. $json = array(
  3472. "apikey" => "FBE7B62621F4CAD7",
  3473. "userkey" => "328BB46EB1E9A0F5",
  3474. "username" => "causefx"
  3475. );
  3476. $api = curl_post("https://api.thetvdb.com/login", $json, $headers);
  3477. return json_decode($api['content'], true)['token'];
  3478. }
  3479. function tvdbGet($id){
  3480. $headers = array(
  3481. "Accept" => "application/json",
  3482. "Authorization" => "Bearer ".tvdbToken(),
  3483. "trakt-api-key" => "4502cfdf8f7282fe454878ff8583f5636392cdc5fcac30d0cc4565f7173bf443",
  3484. "trakt-api-version" => "2"
  3485. );
  3486. $trakt = curl_get("https://api.trakt.tv/search/tvdb/$id?type=show", $headers);
  3487. @$api['trakt'] = json_decode($trakt, true)[0]['show']['ids'];
  3488. if(empty($api['trakt'])){
  3489. $series = curl_get("https://api.thetvdb.com/series/$id", $headers);
  3490. $poster = curl_get("https://api.thetvdb.com/series/$id/images/query?keyType=poster", $headers);
  3491. $backdrop = curl_get("https://api.thetvdb.com/series/$id/images/query?keyType=fanart", $headers);
  3492. $api['series'] = json_decode($series, true)['data'];
  3493. $api['poster'] = json_decode($poster, true)['data'];
  3494. $api['backdrop'] = json_decode($backdrop, true)['data'];
  3495. }
  3496. return $api;
  3497. }
  3498. function tvdbSearch($name, $type){
  3499. $name = rawurlencode(preg_replace("/\(([^()]*+|(?R))*\)/","", $name));
  3500. $headers = array(
  3501. "Accept" => "application/json",
  3502. "Authorization" => "Bearer ".tvdbToken(),
  3503. "trakt-api-key" => "4502cfdf8f7282fe454878ff8583f5636392cdc5fcac30d0cc4565f7173bf443",
  3504. "trakt-api-version" => "2"
  3505. );
  3506. $trakt = curl_get("https://api.trakt.tv/search/$type?query=$name", $headers);
  3507. @$api['trakt'] = json_decode($trakt, true)[0][$type]['ids'];
  3508. return $api;
  3509. }
  3510. function getPlexPlaylists(){
  3511. $address = qualifyURL(PLEXURL);
  3512. // Perform API requests
  3513. $api = @curl_get($address."/playlists?X-Plex-Token=".PLEXTOKEN);
  3514. libxml_use_internal_errors(true);
  3515. $api = simplexml_load_string($api);
  3516. if (is_array($api) || is_object($api)){
  3517. if (!$api->head->title){
  3518. $getServer = simplexml_load_string(@curl_get($address."/?X-Plex-Token=".PLEXTOKEN));
  3519. if (!$getServer) { return 'Could not load!'; }
  3520. // Identify the local machine
  3521. $gotServer = $getServer['machineIdentifier'];
  3522. $output = "";
  3523. $hideMenu = '<div class="pull-right"><div class="btn-group" role="group"><button type="button" id="playlist-Name" class="btn waves btn-default btn-sm dropdown-toggle waves-effect waves-float" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">Choose A Playlist &nbsp;<span class="caret"></span></button><ul style="right:0; left: auto; height: 200px;" class="dropdown-menu filter-recent-playlist playlist-listing">';
  3524. foreach($api AS $child) {
  3525. $items = array();
  3526. if ($child['playlistType'] == "video" && strpos(strtolower($child['title']) , 'private') === false){
  3527. $api = @curl_get($address.$child['key']."?X-Plex-Token=".PLEXTOKEN);
  3528. $api = simplexml_load_string($api);
  3529. if (is_array($api) || is_object($api)){
  3530. if (!$api->head->title){
  3531. $className = preg_replace("/(\W)+/", "", $api['title']);
  3532. $hideMenu .= '<li data-filter="playlist-'.$className.'" data-name="'.$api['title'].'"><a class="js-filter-'.$className.'" href="javascript:void(0)">'.$api['title'].'</a></li>';
  3533. foreach($api->Video AS $child){
  3534. $items[] = resolvePlexItem($gotServer, PLEXTOKEN, $child, false, false,false,$className);
  3535. }
  3536. if (count($items)) {
  3537. $output .= ''.implode('',$items).'';
  3538. }
  3539. }
  3540. }
  3541. }
  3542. }
  3543. $hideMenu .= '</ul></div></div>';
  3544. return '<div id="playlist-all" class="content-box box-shadow big-box"><h5 id="playlist-title" style="margin-bottom: -20px" class="text-center">All Playlists</h5><div class="recentHeader inbox-pagination all">'.$hideMenu.'</div><br/><br/><div class="recentItems-playlists" data-name="all">'.$output.'</div></div>';
  3545. }else{
  3546. writeLog("error", "PLEX PLAYLIST ERROR: could not connect - check token - if HTTPS, is cert valid");
  3547. }
  3548. }else{
  3549. writeLog("error", "PLEX PLAYLIST ERROR: could not connect - check URL - if HTTPS, is cert valid");
  3550. }
  3551. }
  3552. function readExternalLog($type,$filename,$name = null){
  3553. $log = file($filename, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES);
  3554. $log = array_reverse($log);
  3555. foreach($log as $line){
  3556. if(!empty($line) && $line[0] != " "){
  3557. $line = strip($line);
  3558. if($type == "single"){
  3559. if( strpos( strtolower($line), "ror" ) !== false ) {
  3560. echo "<tr><td class='red-bg'>".$line."</td></tr>";
  3561. }else{
  3562. echo "<tr><td>".$line."</td></tr>";
  3563. }
  3564. }elseif($type == "all"){
  3565. if( strpos( strtolower($line), "ror" ) !== false ) {
  3566. echo "<tr><td class='red-bg'>".$name."</td>";
  3567. echo "<td class='red-bg'>".$line."</td></tr>";
  3568. }else{
  3569. echo "<tr><td>".$name."</td>";
  3570. echo "<td>".$line."</td></tr>";
  3571. }
  3572. }
  3573. }
  3574. }
  3575. }
  3576. function getLogs(){
  3577. $path = __DIR__ ."/logs/";
  3578. @mkdir($path, 0770, true);
  3579. $logs = array();
  3580. $files = array_diff(scandir($path), array('.', '..'));
  3581. foreach($files as $v){
  3582. $title = explode(".", $v)[0];
  3583. $logs[$title] = $path.$v;
  3584. }
  3585. return $logs;
  3586. }
  3587. function getBackups(){
  3588. $path = DATABASE_LOCATION ."backups/";
  3589. @mkdir($path, 0770, true);
  3590. $backups = array();
  3591. $files = array_diff(scandir($path), array('.', '..'));
  3592. return array_reverse($files);
  3593. }
  3594. function getExtension($string) {
  3595. return preg_replace("#(.+)?\.(\w+)(\?.+)?#", "$2", $string);
  3596. }
  3597. function showFile(){
  3598. $file = $_GET['file'];
  3599. $fileType = getExtension($file);
  3600. if($fileType != 'php'){
  3601. header("Content-type: ".mimeTypes()[$fileType]);
  3602. @readfile($file);
  3603. }
  3604. }
  3605. function getCalendar(){
  3606. $sonarr = new Sonarr(SONARRURL, SONARRKEY);
  3607. $radarr = new Sonarr(RADARRURL, RADARRKEY);
  3608. $sickrage = new SickRage(SICKRAGEURL, SICKRAGEKEY);
  3609. $startDate = date('Y-m-d',strtotime("-".CALENDARSTARTDAY." days"));
  3610. $endDate = date('Y-m-d',strtotime("+".CALENDARENDDAY." days"));
  3611. $calendarItems = array();
  3612. if (SONARRURL != "" && qualifyUser(SONARRHOMEAUTH)){
  3613. try {
  3614. $sonarrCalendar = getSonarrCalendar($sonarr->getCalendar($startDate, $endDate));
  3615. if(!empty($sonarrCalendar)) { $calendarItems = array_merge($calendarItems, $sonarrCalendar); }
  3616. } catch (Exception $e) {
  3617. writeLog("error", "SONARR ERROR: ".strip($e->getMessage()));
  3618. }
  3619. }
  3620. if (RADARRURL != "" && qualifyUser(RADARRHOMEAUTH)){
  3621. try {
  3622. $radarrCalendar = getRadarrCalendar($radarr->getCalendar($startDate, $endDate));
  3623. if(!empty($radarrCalendar)) { $calendarItems = array_merge($calendarItems, $radarrCalendar); }
  3624. } catch (Exception $e) {
  3625. writeLog("error", "RADARR ERROR: ".strip($e->getMessage()));
  3626. }
  3627. }
  3628. if (COUCHURL != "" && qualifyUser(COUCHHOMEAUTH)){
  3629. $couchCalendar = getCouchCalendar();
  3630. if(!empty($couchCalendar)) { $calendarItems = array_merge($calendarItems, $couchCalendar); }
  3631. }
  3632. if (HEADPHONESURL != "" && qualifyUser(HEADPHONESHOMEAUTH)){
  3633. $headphonesHistory = getHeadphonesCalendar(HEADPHONESURL, HEADPHONESKEY, "getHistory");
  3634. $headphonesWanted = getHeadphonesCalendar(HEADPHONESURL, HEADPHONESKEY, "getWanted");
  3635. if(!empty($headphonesHistory)) { $calendarItems = array_merge($calendarItems, $headphonesHistory); }
  3636. if(!empty($headphonesWanted)) { $calendarItems = array_merge($calendarItems, $headphonesWanted); }
  3637. }
  3638. if (SICKRAGEURL != "" && qualifyUser(SICKRAGEHOMEAUTH)){
  3639. try {
  3640. $sickrageFuture = getSickrageCalendarWanted($sickrage->future());
  3641. if(!empty($sickrageFuture)) { $calendarItems = array_merge($calendarItems, $sickrageFuture); }
  3642. } catch (Exception $e) {
  3643. writeLog("error", "SICKRAGE/BEARD ERROR: ".strip($e->getMessage()));
  3644. } try {
  3645. $sickrageHistory = getSickrageCalendarHistory($sickrage->history("100","downloaded"));
  3646. if(!empty($sickrageHistory)) { $calendarItems = array_merge($calendarItems, $sickrageHistory); }
  3647. } catch (Exception $e) {
  3648. writeLog("error", "SICKRAGE/BEARD ERROR: ".strip($e->getMessage()));
  3649. }
  3650. }
  3651. return $calendarItems;
  3652. }
  3653. function localURL($url){
  3654. if (strpos($url, 'https') !== false) {
  3655. preg_match("/\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}/", $url, $result);
  3656. $result = (!empty($result) ? true : false);
  3657. return $result;
  3658. }
  3659. }
  3660. function fileArray($files){
  3661. foreach($files as $file){
  3662. if(file_exists($file)){
  3663. $list[] = $file;
  3664. }
  3665. }
  3666. if(!empty($list)){ return $list; }
  3667. }
  3668. function backupDB(){
  3669. if (extension_loaded('ZIP')) {
  3670. $directory = DATABASE_LOCATION."backups/";
  3671. @mkdir($directory, 0770, true);
  3672. $orgFiles = array(
  3673. 'css' => 'custom.css',
  3674. 'temp' => 'cus.sd',
  3675. 'orgLog' => 'org.log',
  3676. 'loginLog' => 'loginLog.json',
  3677. 'chatDB' => 'chatpack.db',
  3678. 'config' => 'config/config.php',
  3679. 'database' => DATABASE_LOCATION.'users.db'
  3680. );
  3681. $files = fileArray($orgFiles);
  3682. if(!empty($files)){
  3683. writeLog("success", "BACKUP: backup process started");
  3684. $zipname = $directory.'backup['.date('Y-m-d_H-i').']['.INSTALLEDVERSION.'].zip';
  3685. $zip = new ZipArchive;
  3686. $zip->open($zipname, ZipArchive::CREATE);
  3687. foreach ($files as $file) {
  3688. $zip->addFile($file);
  3689. }
  3690. $zip->close();
  3691. writeLog("success", "BACKUP: backup process finished");
  3692. return true;
  3693. }else{
  3694. return false;
  3695. }
  3696. }else{
  3697. return false;
  3698. }
  3699. }
  3700. class Ping {
  3701. private $host;
  3702. private $ttl;
  3703. private $timeout;
  3704. private $port = 80;
  3705. private $data = 'Ping';
  3706. private $commandOutput;
  3707. /**
  3708. * Called when the Ping object is created.
  3709. *
  3710. * @param string $host
  3711. * The host to be pinged.
  3712. * @param int $ttl
  3713. * Time-to-live (TTL) (You may get a 'Time to live exceeded' error if this
  3714. * value is set too low. The TTL value indicates the scope or range in which
  3715. * a packet may be forwarded. By convention:
  3716. * - 0 = same host
  3717. * - 1 = same subnet
  3718. * - 32 = same site
  3719. * - 64 = same region
  3720. * - 128 = same continent
  3721. * - 255 = unrestricted
  3722. * @param int $timeout
  3723. * Timeout (in seconds) used for ping and fsockopen().
  3724. * @throws \Exception if the host is not set.
  3725. */
  3726. public function __construct($host, $ttl = 255, $timeout = 10) {
  3727. if (!isset($host)) {
  3728. throw new \Exception("Error: Host name not supplied.");
  3729. }
  3730. $this->host = $host;
  3731. $this->ttl = $ttl;
  3732. $this->timeout = $timeout;
  3733. }
  3734. /**
  3735. * Set the ttl (in hops).
  3736. *
  3737. * @param int $ttl
  3738. * TTL in hops.
  3739. */
  3740. public function setTtl($ttl) {
  3741. $this->ttl = $ttl;
  3742. }
  3743. /**
  3744. * Get the ttl.
  3745. *
  3746. * @return int
  3747. * The current ttl for Ping.
  3748. */
  3749. public function getTtl() {
  3750. return $this->ttl;
  3751. }
  3752. /**
  3753. * Set the timeout.
  3754. *
  3755. * @param int $timeout
  3756. * Time to wait in seconds.
  3757. */
  3758. public function setTimeout($timeout) {
  3759. $this->timeout = $timeout;
  3760. }
  3761. /**
  3762. * Get the timeout.
  3763. *
  3764. * @return int
  3765. * Current timeout for Ping.
  3766. */
  3767. public function getTimeout() {
  3768. return $this->timeout;
  3769. }
  3770. /**
  3771. * Set the host.
  3772. *
  3773. * @param string $host
  3774. * Host name or IP address.
  3775. */
  3776. public function setHost($host) {
  3777. $this->host = $host;
  3778. }
  3779. /**
  3780. * Get the host.
  3781. *
  3782. * @return string
  3783. * The current hostname for Ping.
  3784. */
  3785. public function getHost() {
  3786. return $this->host;
  3787. }
  3788. /**
  3789. * Set the port (only used for fsockopen method).
  3790. *
  3791. * Since regular pings use ICMP and don't need to worry about the concept of
  3792. * 'ports', this is only used for the fsockopen method, which pings servers by
  3793. * checking port 80 (by default).
  3794. *
  3795. * @param int $port
  3796. * Port to use for fsockopen ping (defaults to 80 if not set).
  3797. */
  3798. public function setPort($port) {
  3799. $this->port = $port;
  3800. }
  3801. /**
  3802. * Get the port (only used for fsockopen method).
  3803. *
  3804. * @return int
  3805. * The port used by fsockopen pings.
  3806. */
  3807. public function getPort() {
  3808. return $this->port;
  3809. }
  3810. /**
  3811. * Return the command output when method=exec.
  3812. * @return string
  3813. */
  3814. public function getCommandOutput(){
  3815. return $this->commandOutput;
  3816. }
  3817. /**
  3818. * Matches an IP on command output and returns.
  3819. * @return string
  3820. */
  3821. public function getIpAddress() {
  3822. $out = array();
  3823. if (preg_match('/\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}/', $this->commandOutput, $out)){
  3824. return $out[0];
  3825. }
  3826. return null;
  3827. }
  3828. /**
  3829. * Ping a host.
  3830. *
  3831. * @param string $method
  3832. * Method to use when pinging:
  3833. * - exec (default): Pings through the system ping command. Fast and
  3834. * robust, but a security risk if you pass through user-submitted data.
  3835. * - fsockopen: Pings a server on port 80.
  3836. * - socket: Creates a RAW network socket. Only usable in some
  3837. * environments, as creating a SOCK_RAW socket requires root privileges.
  3838. *
  3839. * @throws InvalidArgumentException if $method is not supported.
  3840. *
  3841. * @return mixed
  3842. * Latency as integer, in ms, if host is reachable or FALSE if host is down.
  3843. */
  3844. public function ping($method = 'exec') {
  3845. $latency = false;
  3846. switch ($method) {
  3847. case 'exec':
  3848. $latency = $this->pingExec();
  3849. break;
  3850. case 'fsockopen':
  3851. $latency = $this->pingFsockopen();
  3852. break;
  3853. case 'socket':
  3854. $latency = $this->pingSocket();
  3855. break;
  3856. default:
  3857. throw new \InvalidArgumentException('Unsupported ping method.');
  3858. }
  3859. // Return the latency.
  3860. return $latency;
  3861. }
  3862. /**
  3863. * The exec method uses the possibly insecure exec() function, which passes
  3864. * the input to the system. This is potentially VERY dangerous if you pass in
  3865. * any user-submitted data. Be SURE you sanitize your inputs!
  3866. *
  3867. * @return int
  3868. * Latency, in ms.
  3869. */
  3870. private function pingExec() {
  3871. $latency = false;
  3872. $ttl = escapeshellcmd($this->ttl);
  3873. $timeout = escapeshellcmd($this->timeout);
  3874. $host = escapeshellcmd($this->host);
  3875. // Exec string for Windows-based systems.
  3876. if (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN') {
  3877. // -n = number of pings; -i = ttl; -w = timeout (in milliseconds).
  3878. $exec_string = 'ping -n 1 -i ' . $ttl . ' -w ' . ($timeout * 1000) . ' ' . $host;
  3879. }
  3880. // Exec string for Darwin based systems (OS X).
  3881. else if(strtoupper(PHP_OS) === 'DARWIN') {
  3882. // -n = numeric output; -c = number of pings; -m = ttl; -t = timeout.
  3883. $exec_string = 'ping -n -c 1 -m ' . $ttl . ' -t ' . $timeout . ' ' . $host;
  3884. }
  3885. // Exec string for other UNIX-based systems (Linux).
  3886. else {
  3887. // -n = numeric output; -c = number of pings; -t = ttl; -W = timeout
  3888. $exec_string = 'ping -n -c 1 -t ' . $ttl . ' -W ' . $timeout . ' ' . $host . ' 2>&1';
  3889. }
  3890. exec($exec_string, $output, $return);
  3891. // Strip empty lines and reorder the indexes from 0 (to make results more
  3892. // uniform across OS versions).
  3893. $this->commandOutput = implode($output, '');
  3894. $output = array_values(array_filter($output));
  3895. // If the result line in the output is not empty, parse it.
  3896. if (!empty($output[1])) {
  3897. // Search for a 'time' value in the result line.
  3898. $response = preg_match("/time(?:=|<)(?<time>[\.0-9]+)(?:|\s)ms/", $output[1], $matches);
  3899. // If there's a result and it's greater than 0, return the latency.
  3900. if ($response > 0 && isset($matches['time'])) {
  3901. $latency = round($matches['time'], 2);
  3902. }
  3903. }
  3904. return $latency;
  3905. }
  3906. /**
  3907. * The fsockopen method simply tries to reach the host on a port. This method
  3908. * is often the fastest, but not necessarily the most reliable. Even if a host
  3909. * doesn't respond, fsockopen may still make a connection.
  3910. *
  3911. * @return int
  3912. * Latency, in ms.
  3913. */
  3914. private function pingFsockopen() {
  3915. $start = microtime(true);
  3916. // fsockopen prints a bunch of errors if a host is unreachable. Hide those
  3917. // irrelevant errors and deal with the results instead.
  3918. $fp = @fsockopen($this->host, $this->port, $errno, $errstr, $this->timeout);
  3919. if (!$fp) {
  3920. $latency = false;
  3921. }
  3922. else {
  3923. $latency = microtime(true) - $start;
  3924. $latency = round($latency * 1000, 2);
  3925. }
  3926. return $latency;
  3927. }
  3928. /**
  3929. * The socket method uses raw network packet data to try sending an ICMP ping
  3930. * packet to a server, then measures the response time. Using this method
  3931. * requires the script to be run with root privileges, though, so this method
  3932. * only works reliably on Windows systems and on Linux servers where the
  3933. * script is not being run as a web user.
  3934. *
  3935. * @return int
  3936. * Latency, in ms.
  3937. */
  3938. private function pingSocket() {
  3939. // Create a package.
  3940. $type = "\x08";
  3941. $code = "\x00";
  3942. $checksum = "\x00\x00";
  3943. $identifier = "\x00\x00";
  3944. $seq_number = "\x00\x00";
  3945. $package = $type . $code . $checksum . $identifier . $seq_number . $this->data;
  3946. // Calculate the checksum.
  3947. $checksum = $this->calculateChecksum($package);
  3948. // Finalize the package.
  3949. $package = $type . $code . $checksum . $identifier . $seq_number . $this->data;
  3950. // Create a socket, connect to server, then read socket and calculate.
  3951. if ($socket = socket_create(AF_INET, SOCK_RAW, 1)) {
  3952. socket_set_option($socket, SOL_SOCKET, SO_RCVTIMEO, array(
  3953. 'sec' => 10,
  3954. 'usec' => 0,
  3955. ));
  3956. // Prevent errors from being printed when host is unreachable.
  3957. @socket_connect($socket, $this->host, null);
  3958. $start = microtime(true);
  3959. // Send the package.
  3960. @socket_send($socket, $package, strlen($package), 0);
  3961. if (socket_read($socket, 255) !== false) {
  3962. $latency = microtime(true) - $start;
  3963. $latency = round($latency * 1000, 2);
  3964. }
  3965. else {
  3966. $latency = false;
  3967. }
  3968. }
  3969. else {
  3970. $latency = false;
  3971. }
  3972. // Close the socket.
  3973. socket_close($socket);
  3974. return $latency;
  3975. }
  3976. /**
  3977. * Calculate a checksum.
  3978. *
  3979. * @param string $data
  3980. * Data for which checksum will be calculated.
  3981. *
  3982. * @return string
  3983. * Binary string checksum of $data.
  3984. */
  3985. private function calculateChecksum($data) {
  3986. if (strlen($data) % 2) {
  3987. $data .= "\x00";
  3988. }
  3989. $bit = unpack('n*', $data);
  3990. $sum = array_sum($bit);
  3991. while ($sum >> 16) {
  3992. $sum = ($sum >> 16) + ($sum & 0xffff);
  3993. }
  3994. return pack('n*', ~$sum);
  3995. }
  3996. }
  3997. function ping($pings, $type = "string") {
  3998. $ping = new Ping("");
  3999. $ping->setTtl(128);
  4000. $ping->setTimeout(2);
  4001. switch ($type){
  4002. case "array":
  4003. $results = [];
  4004. foreach ($pings as $k => $v) {
  4005. if(strpos($v, ':') !== false){
  4006. $domain = explode(':', $v)[0];
  4007. $port = explode(':', $v)[1];
  4008. $ping->setHost($domain);
  4009. $ping->setPort($port);
  4010. $latency = $ping->ping('fsockopen');
  4011. }else{
  4012. $ping->setHost($v);
  4013. $latency = $ping->ping();
  4014. }
  4015. if ($latency || $latency === 0) {
  4016. $results[$k] = $latency;
  4017. } else {
  4018. $results[$k] = 0;
  4019. }
  4020. }
  4021. break;
  4022. case "string":
  4023. if(strpos($pings, ':') !== false){
  4024. $domain = explode(':', $pings)[0];
  4025. $port = explode(':', $pings)[1];
  4026. $ping->setHost($domain);
  4027. $ping->setPort($port);
  4028. $latency = $ping->ping('fsockopen');
  4029. }else{
  4030. $ping->setHost($pings);
  4031. $latency = $ping->ping();
  4032. }
  4033. if ($latency || $latency === 0) {
  4034. $results = $latency;
  4035. } else {
  4036. $results = 0;
  4037. }
  4038. break;
  4039. }
  4040. return $results;
  4041. }
  4042. function getPing($url, $style, $refresh = null){
  4043. if(ping($url) !== 0){
  4044. $class = 'success';
  4045. if(!$refresh){
  4046. $class .= " animated slideInLeft";
  4047. }
  4048. }else{
  4049. $class = "warning";
  4050. if(!$refresh){
  4051. $class .= " animated flash loop-animation-timeout";
  4052. }
  4053. }
  4054. echo '<span class="badge ping-'.$class.'" style="position: absolute;z-index: 100;right: 5px; padding: 0px 0px;'.$style.';font-size: 10px;">&nbsp;</span>';
  4055. }
  4056. function speedTestData(){
  4057. $file_db = DATABASE_LOCATION."speedtest.db";
  4058. if(file_exists($file_db)){
  4059. $conn = new PDO("sqlite:$file_db") or die("1");
  4060. $result = $conn->query('SELECT * FROM speedtest_users');
  4061. $conn = null;
  4062. if (is_array($result) || is_object($result)){
  4063. foreach($result as $k => $v){
  4064. $return[$k] = $v;
  4065. }
  4066. return $return;
  4067. }
  4068. }
  4069. }
  4070. function speedTestDisplay($array, $output){
  4071. if (is_array($array) || is_object($array)){
  4072. if($output == "graph"){
  4073. $result = "Morris.Line({element: 'morris-line',data: [";
  4074. foreach($array as $k => $v){
  4075. $result .= "{ y: '".substr($v['timestamp'],0,10)."', a: ".$v['ul'].", b: ".$v['dl'].", c: ".$v['ping']." },";
  4076. }
  4077. $result .= "],xkey: 'y',ykeys: ['a', 'b', 'c'],labels: ['Upload', 'Download', 'Ping'],hideHover: 'auto',resize: true,lineColors: ['#63A8EB','#ccc','#000'] });";
  4078. }elseif($output == "table"){
  4079. $result = "";
  4080. foreach($array as $k => $v){
  4081. $result .= "<tr><td>".$v['timestamp']."</td><td>".$v['ip']."</td><td>".$v['dl']."</td><td>".$v['ul']."</td><td>".$v['ping']."</td><td>".$v['jitter']."</td></tr>";
  4082. }
  4083. }
  4084. return $result;
  4085. }
  4086. }
  4087. function buildMenuPhone($array){
  4088. if (is_array($array) || is_object($array)){
  4089. $result = '
  4090. <div class="content-box profile-sidebar box-shadow">
  4091. <img src="images/organizr-logo-h-d.png" width="100%" style="margin-top: -10px;">
  4092. <div class="profile-usermenu">
  4093. <ul class="nav" id="settings-list">
  4094. ';
  4095. foreach($array as $k => $v){
  4096. if($v['id'] == 'open-invites' && empty(PLEXURL)){
  4097. continue;
  4098. }
  4099. if($v['id'] == 'open-email' && ENABLEMAIL !== "true"){
  4100. continue;
  4101. }
  4102. /*$result .= '
  4103. <li>
  4104. <a id="'.$v['id'].'" box="'.$v['box'].'">'.$v['name'].'
  4105. <span class="fa-stack fa-fw pull-right" style="margin-top: -5px;margin-right: -10px;">
  4106. <i class="fa fa-'.$v['icon_1'].' fa-stack-2x '.$v['color'].'" style="font-size:null;"></i>
  4107. <i class="fa fa-'.$v['icon_2'].' fa-stack-1x fa-inverse"></i>
  4108. </span>
  4109. </a>
  4110. </li>
  4111. ';*/
  4112. $result .= '<li><a id="'.$v['id'].'" box="'.$v['box'].'"><i class="fa fa-'.$v['icon_2'].' '.$v['color'].' fa-fw pull-right"></i>'.$v['name'].'</a></li>';
  4113. }
  4114. $result .= '</ul></div></div>';
  4115. return $result;
  4116. }
  4117. }
  4118. function buildMenu($array){
  4119. if (is_array($array) || is_object($array)){
  4120. $result = '<div class="settingsList">';
  4121. foreach($array as $k => $v){
  4122. if($v['id'] == 'open-invites' && empty(PLEXURL)){
  4123. continue;
  4124. }
  4125. if($v['id'] == 'open-email' && ENABLEMAIL !== "true"){
  4126. continue;
  4127. }
  4128. $result .= '
  4129. <button id="'.$v['id'].'" box="'.$v['box'].'" type="button" style="border-radius: 0px !important; -webkit-border-radius: 0px !important;margin-bottom: 3px;margin-left:5px;color:white;" class="btn '.$v['color2'].' btn-icon waves waves-circle waves-effect waves-float settingsMenu">
  4130. <i class="mdi mdi-'.$v['icon_1'].' fa-fw pull-left" style="padding-left: '.$v['padding'].'px;font-size: 30px"></i>
  4131. <p class="" style="text-align: center;direction: rtl;display:none;margin: 2px;"><strong>'.$v['name'].'</strong></p>
  4132. </button>
  4133. ';
  4134. }
  4135. $result .= '</div>';
  4136. return $result;
  4137. }
  4138. }
  4139. function requestInvite($email, $username){
  4140. //sendEmail($email, $username = "Organizr User", $subject, $body, $cc = null){
  4141. //orgEmail($header = "Message From Admin", $title = "Important Message", $user = "Organizr User", $mainMessage = "", $button = null, $buttonURL = null, $subTitle = "", $subMessage = ""){
  4142. sendEmail($GLOBALS['USER']->adminEmail, "Admin", "Plex Invite Request", orgEmail("PLEX Invite Request", "Look who wants to join the cool club", "Admin", "Hey, The User: $user has requested access to your Plex Library.", "Generate Invite", null, "What Next?", "Well, That is up to you. You can go check on them if you like."));
  4143. }
  4144. function errormessage($msg) {
  4145. echo "<div style=\"margin-top: 50px;\">";
  4146. echo "<span style=\"color:#d89334;\">error </span>";
  4147. echo $msg;
  4148. echo "</div>";
  4149. }
  4150. function getOrgUsers(){
  4151. $file_db = DATABASE_LOCATION."users.db";
  4152. if(file_exists($file_db)){
  4153. $conn = new PDO("sqlite:$file_db") or die("1");
  4154. $result = $conn->query('SELECT * FROM users');
  4155. $conn = null;
  4156. if (is_array($result) || is_object($result)){
  4157. foreach($result as $k => $v){
  4158. $return[$v['username']] = $v['email'];
  4159. }
  4160. return $return;
  4161. }
  4162. }
  4163. }
  4164. function getEmails($type = 'org'){
  4165. if($type == 'plex'){
  4166. $emails = array_merge(libraryList()['both'],getOrgUsers());
  4167. }elseif($type == 'emby'){
  4168. $emails = getOrgUsers();
  4169. }else{
  4170. $emails = getOrgUsers();
  4171. }
  4172. return $emails;
  4173. }
  4174. function printEmails($emails){
  4175. $result = '';
  4176. foreach($emails as $k => $v){
  4177. $result .= '<option value="'.$v.'">'.$k.'</option>';
  4178. }
  4179. return $result;
  4180. }
  4181. function massEmail($to, $subject, $message){
  4182. if (!isset($GLOBALS['file_db'])) {
  4183. $GLOBALS['file_db'] = new PDO('sqlite:'.DATABASE_LOCATION.'users.db');
  4184. $GLOBALS['file_db']->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
  4185. }
  4186. sendEmail(null, null, $subject, orgEmail("Message From Admin", "Important Information", "There", $message, null, null, "Thank You!", "Thanks for taking the time to read this message from me."),$GLOBALS['USER']->adminEmail,$to);
  4187. }
  4188. function q2a($q){
  4189. if (is_array($q) || is_object($q)){
  4190. foreach ($q as $k => $v){
  4191. $a[$k] = $v;
  4192. }
  4193. if(!empty($a)){
  4194. return $a;
  4195. }
  4196. }
  4197. }
  4198. class Mobile_Detect
  4199. {
  4200. /**
  4201. * Mobile detection type.
  4202. *
  4203. * @deprecated since version 2.6.9
  4204. */
  4205. const DETECTION_TYPE_MOBILE = 'mobile';
  4206. /**
  4207. * Extended detection type.
  4208. *
  4209. * @deprecated since version 2.6.9
  4210. */
  4211. const DETECTION_TYPE_EXTENDED = 'extended';
  4212. /**
  4213. * A frequently used regular expression to extract version #s.
  4214. *
  4215. * @deprecated since version 2.6.9
  4216. */
  4217. const VER = '([\w._\+]+)';
  4218. /**
  4219. * Top-level device.
  4220. */
  4221. const MOBILE_GRADE_A = 'A';
  4222. /**
  4223. * Mid-level device.
  4224. */
  4225. const MOBILE_GRADE_B = 'B';
  4226. /**
  4227. * Low-level device.
  4228. */
  4229. const MOBILE_GRADE_C = 'C';
  4230. /**
  4231. * Stores the version number of the current release.
  4232. */
  4233. const VERSION = '2.8.26';
  4234. /**
  4235. * A type for the version() method indicating a string return value.
  4236. */
  4237. const VERSION_TYPE_STRING = 'text';
  4238. /**
  4239. * A type for the version() method indicating a float return value.
  4240. */
  4241. const VERSION_TYPE_FLOAT = 'float';
  4242. /**
  4243. * A cache for resolved matches
  4244. * @var array
  4245. */
  4246. protected $cache = array();
  4247. /**
  4248. * The User-Agent HTTP header is stored in here.
  4249. * @var string
  4250. */
  4251. protected $userAgent = null;
  4252. /**
  4253. * HTTP headers in the PHP-flavor. So HTTP_USER_AGENT and SERVER_SOFTWARE.
  4254. * @var array
  4255. */
  4256. protected $httpHeaders = array();
  4257. /**
  4258. * CloudFront headers. E.g. CloudFront-Is-Desktop-Viewer, CloudFront-Is-Mobile-Viewer & CloudFront-Is-Tablet-Viewer.
  4259. * @var array
  4260. */
  4261. protected $cloudfrontHeaders = array();
  4262. /**
  4263. * The matching Regex.
  4264. * This is good for debug.
  4265. * @var string
  4266. */
  4267. protected $matchingRegex = null;
  4268. /**
  4269. * The matches extracted from the regex expression.
  4270. * This is good for debug.
  4271. * @var string
  4272. */
  4273. protected $matchesArray = null;
  4274. /**
  4275. * The detection type, using self::DETECTION_TYPE_MOBILE or self::DETECTION_TYPE_EXTENDED.
  4276. *
  4277. * @deprecated since version 2.6.9
  4278. *
  4279. * @var string
  4280. */
  4281. protected $detectionType = self::DETECTION_TYPE_MOBILE;
  4282. /**
  4283. * HTTP headers that trigger the 'isMobile' detection
  4284. * to be true.
  4285. *
  4286. * @var array
  4287. */
  4288. protected static $mobileHeaders = array(
  4289. 'HTTP_ACCEPT' => array('matches' => array(
  4290. // Opera Mini; @reference: http://dev.opera.com/articles/view/opera-binary-markup-language/
  4291. 'application/x-obml2d',
  4292. // BlackBerry devices.
  4293. 'application/vnd.rim.html',
  4294. 'text/vnd.wap.wml',
  4295. 'application/vnd.wap.xhtml+xml'
  4296. )),
  4297. 'HTTP_X_WAP_PROFILE' => null,
  4298. 'HTTP_X_WAP_CLIENTID' => null,
  4299. 'HTTP_WAP_CONNECTION' => null,
  4300. 'HTTP_PROFILE' => null,
  4301. // Reported by Opera on Nokia devices (eg. C3).
  4302. 'HTTP_X_OPERAMINI_PHONE_UA' => null,
  4303. 'HTTP_X_NOKIA_GATEWAY_ID' => null,
  4304. 'HTTP_X_ORANGE_ID' => null,
  4305. 'HTTP_X_VODAFONE_3GPDPCONTEXT' => null,
  4306. 'HTTP_X_HUAWEI_USERID' => null,
  4307. // Reported by Windows Smartphones.
  4308. 'HTTP_UA_OS' => null,
  4309. // Reported by Verizon, Vodafone proxy system.
  4310. 'HTTP_X_MOBILE_GATEWAY' => null,
  4311. // Seen this on HTC Sensation. SensationXE_Beats_Z715e.
  4312. 'HTTP_X_ATT_DEVICEID' => null,
  4313. // Seen this on a HTC.
  4314. 'HTTP_UA_CPU' => array('matches' => array('ARM')),
  4315. );
  4316. /**
  4317. * List of mobile devices (phones).
  4318. *
  4319. * @var array
  4320. */
  4321. protected static $phoneDevices = array(
  4322. 'iPhone' => '\biPhone\b|\biPod\b', // |\biTunes
  4323. 'BlackBerry' => 'BlackBerry|\bBB10\b|rim[0-9]+',
  4324. 'HTC' => 'HTC|HTC.*(Sensation|Evo|Vision|Explorer|6800|8100|8900|A7272|S510e|C110e|Legend|Desire|T8282)|APX515CKT|Qtek9090|APA9292KT|HD_mini|Sensation.*Z710e|PG86100|Z715e|Desire.*(A8181|HD)|ADR6200|ADR6400L|ADR6425|001HT|Inspire 4G|Android.*\bEVO\b|T-Mobile G1|Z520m',
  4325. 'Nexus' => 'Nexus One|Nexus S|Galaxy.*Nexus|Android.*Nexus.*Mobile|Nexus 4|Nexus 5|Nexus 6',
  4326. // @todo: Is 'Dell Streak' a tablet or a phone? ;)
  4327. 'Dell' => 'Dell.*Streak|Dell.*Aero|Dell.*Venue|DELL.*Venue Pro|Dell Flash|Dell Smoke|Dell Mini 3iX|XCD28|XCD35|\b001DL\b|\b101DL\b|\bGS01\b',
  4328. 'Motorola' => 'Motorola|DROIDX|DROID BIONIC|\bDroid\b.*Build|Android.*Xoom|HRI39|MOT-|A1260|A1680|A555|A853|A855|A953|A955|A956|Motorola.*ELECTRIFY|Motorola.*i1|i867|i940|MB200|MB300|MB501|MB502|MB508|MB511|MB520|MB525|MB526|MB611|MB612|MB632|MB810|MB855|MB860|MB861|MB865|MB870|ME501|ME502|ME511|ME525|ME600|ME632|ME722|ME811|ME860|ME863|ME865|MT620|MT710|MT716|MT720|MT810|MT870|MT917|Motorola.*TITANIUM|WX435|WX445|XT300|XT301|XT311|XT316|XT317|XT319|XT320|XT390|XT502|XT530|XT531|XT532|XT535|XT603|XT610|XT611|XT615|XT681|XT701|XT702|XT711|XT720|XT800|XT806|XT860|XT862|XT875|XT882|XT883|XT894|XT901|XT907|XT909|XT910|XT912|XT928|XT926|XT915|XT919|XT925|XT1021|\bMoto E\b',
  4329. 'Samsung' => '\bSamsung\b|SM-G9250|GT-19300|SGH-I337|BGT-S5230|GT-B2100|GT-B2700|GT-B2710|GT-B3210|GT-B3310|GT-B3410|GT-B3730|GT-B3740|GT-B5510|GT-B5512|GT-B5722|GT-B6520|GT-B7300|GT-B7320|GT-B7330|GT-B7350|GT-B7510|GT-B7722|GT-B7800|GT-C3010|GT-C3011|GT-C3060|GT-C3200|GT-C3212|GT-C3212I|GT-C3262|GT-C3222|GT-C3300|GT-C3300K|GT-C3303|GT-C3303K|GT-C3310|GT-C3322|GT-C3330|GT-C3350|GT-C3500|GT-C3510|GT-C3530|GT-C3630|GT-C3780|GT-C5010|GT-C5212|GT-C6620|GT-C6625|GT-C6712|GT-E1050|GT-E1070|GT-E1075|GT-E1080|GT-E1081|GT-E1085|GT-E1087|GT-E1100|GT-E1107|GT-E1110|GT-E1120|GT-E1125|GT-E1130|GT-E1160|GT-E1170|GT-E1175|GT-E1180|GT-E1182|GT-E1200|GT-E1210|GT-E1225|GT-E1230|GT-E1390|GT-E2100|GT-E2120|GT-E2121|GT-E2152|GT-E2220|GT-E2222|GT-E2230|GT-E2232|GT-E2250|GT-E2370|GT-E2550|GT-E2652|GT-E3210|GT-E3213|GT-I5500|GT-I5503|GT-I5700|GT-I5800|GT-I5801|GT-I6410|GT-I6420|GT-I7110|GT-I7410|GT-I7500|GT-I8000|GT-I8150|GT-I8160|GT-I8190|GT-I8320|GT-I8330|GT-I8350|GT-I8530|GT-I8700|GT-I8703|GT-I8910|GT-I9000|GT-I9001|GT-I9003|GT-I9010|GT-I9020|GT-I9023|GT-I9070|GT-I9082|GT-I9100|GT-I9103|GT-I9220|GT-I9250|GT-I9300|GT-I9305|GT-I9500|GT-I9505|GT-M3510|GT-M5650|GT-M7500|GT-M7600|GT-M7603|GT-M8800|GT-M8910|GT-N7000|GT-S3110|GT-S3310|GT-S3350|GT-S3353|GT-S3370|GT-S3650|GT-S3653|GT-S3770|GT-S3850|GT-S5210|GT-S5220|GT-S5229|GT-S5230|GT-S5233|GT-S5250|GT-S5253|GT-S5260|GT-S5263|GT-S5270|GT-S5300|GT-S5330|GT-S5350|GT-S5360|GT-S5363|GT-S5369|GT-S5380|GT-S5380D|GT-S5560|GT-S5570|GT-S5600|GT-S5603|GT-S5610|GT-S5620|GT-S5660|GT-S5670|GT-S5690|GT-S5750|GT-S5780|GT-S5830|GT-S5839|GT-S6102|GT-S6500|GT-S7070|GT-S7200|GT-S7220|GT-S7230|GT-S7233|GT-S7250|GT-S7500|GT-S7530|GT-S7550|GT-S7562|GT-S7710|GT-S8000|GT-S8003|GT-S8500|GT-S8530|GT-S8600|SCH-A310|SCH-A530|SCH-A570|SCH-A610|SCH-A630|SCH-A650|SCH-A790|SCH-A795|SCH-A850|SCH-A870|SCH-A890|SCH-A930|SCH-A950|SCH-A970|SCH-A990|SCH-I100|SCH-I110|SCH-I400|SCH-I405|SCH-I500|SCH-I510|SCH-I515|SCH-I600|SCH-I730|SCH-I760|SCH-I770|SCH-I830|SCH-I910|SCH-I920|SCH-I959|SCH-LC11|SCH-N150|SCH-N300|SCH-R100|SCH-R300|SCH-R351|SCH-R400|SCH-R410|SCH-T300|SCH-U310|SCH-U320|SCH-U350|SCH-U360|SCH-U365|SCH-U370|SCH-U380|SCH-U410|SCH-U430|SCH-U450|SCH-U460|SCH-U470|SCH-U490|SCH-U540|SCH-U550|SCH-U620|SCH-U640|SCH-U650|SCH-U660|SCH-U700|SCH-U740|SCH-U750|SCH-U810|SCH-U820|SCH-U900|SCH-U940|SCH-U960|SCS-26UC|SGH-A107|SGH-A117|SGH-A127|SGH-A137|SGH-A157|SGH-A167|SGH-A177|SGH-A187|SGH-A197|SGH-A227|SGH-A237|SGH-A257|SGH-A437|SGH-A517|SGH-A597|SGH-A637|SGH-A657|SGH-A667|SGH-A687|SGH-A697|SGH-A707|SGH-A717|SGH-A727|SGH-A737|SGH-A747|SGH-A767|SGH-A777|SGH-A797|SGH-A817|SGH-A827|SGH-A837|SGH-A847|SGH-A867|SGH-A877|SGH-A887|SGH-A897|SGH-A927|SGH-B100|SGH-B130|SGH-B200|SGH-B220|SGH-C100|SGH-C110|SGH-C120|SGH-C130|SGH-C140|SGH-C160|SGH-C170|SGH-C180|SGH-C200|SGH-C207|SGH-C210|SGH-C225|SGH-C230|SGH-C417|SGH-C450|SGH-D307|SGH-D347|SGH-D357|SGH-D407|SGH-D415|SGH-D780|SGH-D807|SGH-D980|SGH-E105|SGH-E200|SGH-E315|SGH-E316|SGH-E317|SGH-E335|SGH-E590|SGH-E635|SGH-E715|SGH-E890|SGH-F300|SGH-F480|SGH-I200|SGH-I300|SGH-I320|SGH-I550|SGH-I577|SGH-I600|SGH-I607|SGH-I617|SGH-I627|SGH-I637|SGH-I677|SGH-I700|SGH-I717|SGH-I727|SGH-i747M|SGH-I777|SGH-I780|SGH-I827|SGH-I847|SGH-I857|SGH-I896|SGH-I897|SGH-I900|SGH-I907|SGH-I917|SGH-I927|SGH-I937|SGH-I997|SGH-J150|SGH-J200|SGH-L170|SGH-L700|SGH-M110|SGH-M150|SGH-M200|SGH-N105|SGH-N500|SGH-N600|SGH-N620|SGH-N625|SGH-N700|SGH-N710|SGH-P107|SGH-P207|SGH-P300|SGH-P310|SGH-P520|SGH-P735|SGH-P777|SGH-Q105|SGH-R210|SGH-R220|SGH-R225|SGH-S105|SGH-S307|SGH-T109|SGH-T119|SGH-T139|SGH-T209|SGH-T219|SGH-T229|SGH-T239|SGH-T249|SGH-T259|SGH-T309|SGH-T319|SGH-T329|SGH-T339|SGH-T349|SGH-T359|SGH-T369|SGH-T379|SGH-T409|SGH-T429|SGH-T439|SGH-T459|SGH-T469|SGH-T479|SGH-T499|SGH-T509|SGH-T519|SGH-T539|SGH-T559|SGH-T589|SGH-T609|SGH-T619|SGH-T629|SGH-T639|SGH-T659|SGH-T669|SGH-T679|SGH-T709|SGH-T719|SGH-T729|SGH-T739|SGH-T746|SGH-T749|SGH-T759|SGH-T769|SGH-T809|SGH-T819|SGH-T839|SGH-T919|SGH-T929|SGH-T939|SGH-T959|SGH-T989|SGH-U100|SGH-U200|SGH-U800|SGH-V205|SGH-V206|SGH-X100|SGH-X105|SGH-X120|SGH-X140|SGH-X426|SGH-X427|SGH-X475|SGH-X495|SGH-X497|SGH-X507|SGH-X600|SGH-X610|SGH-X620|SGH-X630|SGH-X700|SGH-X820|SGH-X890|SGH-Z130|SGH-Z150|SGH-Z170|SGH-ZX10|SGH-ZX20|SHW-M110|SPH-A120|SPH-A400|SPH-A420|SPH-A460|SPH-A500|SPH-A560|SPH-A600|SPH-A620|SPH-A660|SPH-A700|SPH-A740|SPH-A760|SPH-A790|SPH-A800|SPH-A820|SPH-A840|SPH-A880|SPH-A900|SPH-A940|SPH-A960|SPH-D600|SPH-D700|SPH-D710|SPH-D720|SPH-I300|SPH-I325|SPH-I330|SPH-I350|SPH-I500|SPH-I600|SPH-I700|SPH-L700|SPH-M100|SPH-M220|SPH-M240|SPH-M300|SPH-M305|SPH-M320|SPH-M330|SPH-M350|SPH-M360|SPH-M370|SPH-M380|SPH-M510|SPH-M540|SPH-M550|SPH-M560|SPH-M570|SPH-M580|SPH-M610|SPH-M620|SPH-M630|SPH-M800|SPH-M810|SPH-M850|SPH-M900|SPH-M910|SPH-M920|SPH-M930|SPH-N100|SPH-N200|SPH-N240|SPH-N300|SPH-N400|SPH-Z400|SWC-E100|SCH-i909|GT-N7100|GT-N7105|SCH-I535|SM-N900A|SGH-I317|SGH-T999L|GT-S5360B|GT-I8262|GT-S6802|GT-S6312|GT-S6310|GT-S5312|GT-S5310|GT-I9105|GT-I8510|GT-S6790N|SM-G7105|SM-N9005|GT-S5301|GT-I9295|GT-I9195|SM-C101|GT-S7392|GT-S7560|GT-B7610|GT-I5510|GT-S7582|GT-S7530E|GT-I8750|SM-G9006V|SM-G9008V|SM-G9009D|SM-G900A|SM-G900D|SM-G900F|SM-G900H|SM-G900I|SM-G900J|SM-G900K|SM-G900L|SM-G900M|SM-G900P|SM-G900R4|SM-G900S|SM-G900T|SM-G900V|SM-G900W8|SHV-E160K|SCH-P709|SCH-P729|SM-T2558|GT-I9205|SM-G9350|SM-J120F|SM-G920F|SM-G920V|SM-G930F|SM-N910C',
  4330. 'LG' => '\bLG\b;|LG[- ]?(C800|C900|E400|E610|E900|E-900|F160|F180K|F180L|F180S|730|855|L160|LS740|LS840|LS970|LU6200|MS690|MS695|MS770|MS840|MS870|MS910|P500|P700|P705|VM696|AS680|AS695|AX840|C729|E970|GS505|272|C395|E739BK|E960|L55C|L75C|LS696|LS860|P769BK|P350|P500|P509|P870|UN272|US730|VS840|VS950|LN272|LN510|LS670|LS855|LW690|MN270|MN510|P509|P769|P930|UN200|UN270|UN510|UN610|US670|US740|US760|UX265|UX840|VN271|VN530|VS660|VS700|VS740|VS750|VS910|VS920|VS930|VX9200|VX11000|AX840A|LW770|P506|P925|P999|E612|D955|D802|MS323)',
  4331. 'Sony' => 'SonyST|SonyLT|SonyEricsson|SonyEricssonLT15iv|LT18i|E10i|LT28h|LT26w|SonyEricssonMT27i|C5303|C6902|C6903|C6906|C6943|D2533',
  4332. 'Asus' => 'Asus.*Galaxy|PadFone.*Mobile',
  4333. 'NokiaLumia' => 'Lumia [0-9]{3,4}',
  4334. // http://www.micromaxinfo.com/mobiles/smartphones
  4335. // Added because the codes might conflict with Acer Tablets.
  4336. 'Micromax' => 'Micromax.*\b(A210|A92|A88|A72|A111|A110Q|A115|A116|A110|A90S|A26|A51|A35|A54|A25|A27|A89|A68|A65|A57|A90)\b',
  4337. // @todo Complete the regex.
  4338. 'Palm' => 'PalmSource|Palm', // avantgo|blazer|elaine|hiptop|plucker|xiino ;
  4339. 'Vertu' => 'Vertu|Vertu.*Ltd|Vertu.*Ascent|Vertu.*Ayxta|Vertu.*Constellation(F|Quest)?|Vertu.*Monika|Vertu.*Signature', // Just for fun ;)
  4340. // http://www.pantech.co.kr/en/prod/prodList.do?gbrand=VEGA (PANTECH)
  4341. // Most of the VEGA devices are legacy. PANTECH seem to be newer devices based on Android.
  4342. 'Pantech' => 'PANTECH|IM-A850S|IM-A840S|IM-A830L|IM-A830K|IM-A830S|IM-A820L|IM-A810K|IM-A810S|IM-A800S|IM-T100K|IM-A725L|IM-A780L|IM-A775C|IM-A770K|IM-A760S|IM-A750K|IM-A740S|IM-A730S|IM-A720L|IM-A710K|IM-A690L|IM-A690S|IM-A650S|IM-A630K|IM-A600S|VEGA PTL21|PT003|P8010|ADR910L|P6030|P6020|P9070|P4100|P9060|P5000|CDM8992|TXT8045|ADR8995|IS11PT|P2030|P6010|P8000|PT002|IS06|CDM8999|P9050|PT001|TXT8040|P2020|P9020|P2000|P7040|P7000|C790',
  4343. // http://www.fly-phone.com/devices/smartphones/ ; Included only smartphones.
  4344. 'Fly' => 'IQ230|IQ444|IQ450|IQ440|IQ442|IQ441|IQ245|IQ256|IQ236|IQ255|IQ235|IQ245|IQ275|IQ240|IQ285|IQ280|IQ270|IQ260|IQ250',
  4345. // http://fr.wikomobile.com
  4346. 'Wiko' => 'KITE 4G|HIGHWAY|GETAWAY|STAIRWAY|DARKSIDE|DARKFULL|DARKNIGHT|DARKMOON|SLIDE|WAX 4G|RAINBOW|BLOOM|SUNSET|GOA(?!nna)|LENNY|BARRY|IGGY|OZZY|CINK FIVE|CINK PEAX|CINK PEAX 2|CINK SLIM|CINK SLIM 2|CINK +|CINK KING|CINK PEAX|CINK SLIM|SUBLIM',
  4347. 'iMobile' => 'i-mobile (IQ|i-STYLE|idea|ZAA|Hitz)',
  4348. // Added simvalley mobile just for fun. They have some interesting devices.
  4349. // http://www.simvalley.fr/telephonie---gps-_22_telephonie-mobile_telephones_.html
  4350. 'SimValley' => '\b(SP-80|XT-930|SX-340|XT-930|SX-310|SP-360|SP60|SPT-800|SP-120|SPT-800|SP-140|SPX-5|SPX-8|SP-100|SPX-8|SPX-12)\b',
  4351. // Wolfgang - a brand that is sold by Aldi supermarkets.
  4352. // http://www.wolfgangmobile.com/
  4353. 'Wolfgang' => 'AT-B24D|AT-AS50HD|AT-AS40W|AT-AS55HD|AT-AS45q2|AT-B26D|AT-AS50Q',
  4354. 'Alcatel' => 'Alcatel',
  4355. 'Nintendo' => 'Nintendo 3DS',
  4356. // http://en.wikipedia.org/wiki/Amoi
  4357. 'Amoi' => 'Amoi',
  4358. // http://en.wikipedia.org/wiki/INQ
  4359. 'INQ' => 'INQ',
  4360. // @Tapatalk is a mobile app; http://support.tapatalk.com/threads/smf-2-0-2-os-and-browser-detection-plugin-and-tapatalk.15565/#post-79039
  4361. 'GenericPhone' => 'Tapatalk|PDA;|SAGEM|\bmmp\b|pocket|\bpsp\b|symbian|Smartphone|smartfon|treo|up.browser|up.link|vodafone|\bwap\b|nokia|Series40|Series60|S60|SonyEricsson|N900|MAUI.*WAP.*Browser',
  4362. );
  4363. /**
  4364. * List of tablet devices.
  4365. *
  4366. * @var array
  4367. */
  4368. protected static $tabletDevices = array(
  4369. // @todo: check for mobile friendly emails topic.
  4370. 'iPad' => 'iPad|iPad.*Mobile',
  4371. // Removed |^.*Android.*Nexus(?!(?:Mobile).)*$
  4372. // @see #442
  4373. 'NexusTablet' => 'Android.*Nexus[\s]+(7|9|10)',
  4374. 'SamsungTablet' => 'SAMSUNG.*Tablet|Galaxy.*Tab|SC-01C|GT-P1000|GT-P1003|GT-P1010|GT-P3105|GT-P6210|GT-P6800|GT-P6810|GT-P7100|GT-P7300|GT-P7310|GT-P7500|GT-P7510|SCH-I800|SCH-I815|SCH-I905|SGH-I957|SGH-I987|SGH-T849|SGH-T859|SGH-T869|SPH-P100|GT-P3100|GT-P3108|GT-P3110|GT-P5100|GT-P5110|GT-P6200|GT-P7320|GT-P7511|GT-N8000|GT-P8510|SGH-I497|SPH-P500|SGH-T779|SCH-I705|SCH-I915|GT-N8013|GT-P3113|GT-P5113|GT-P8110|GT-N8010|GT-N8005|GT-N8020|GT-P1013|GT-P6201|GT-P7501|GT-N5100|GT-N5105|GT-N5110|SHV-E140K|SHV-E140L|SHV-E140S|SHV-E150S|SHV-E230K|SHV-E230L|SHV-E230S|SHW-M180K|SHW-M180L|SHW-M180S|SHW-M180W|SHW-M300W|SHW-M305W|SHW-M380K|SHW-M380S|SHW-M380W|SHW-M430W|SHW-M480K|SHW-M480S|SHW-M480W|SHW-M485W|SHW-M486W|SHW-M500W|GT-I9228|SCH-P739|SCH-I925|GT-I9200|GT-P5200|GT-P5210|GT-P5210X|SM-T311|SM-T310|SM-T310X|SM-T210|SM-T210R|SM-T211|SM-P600|SM-P601|SM-P605|SM-P900|SM-P901|SM-T217|SM-T217A|SM-T217S|SM-P6000|SM-T3100|SGH-I467|XE500|SM-T110|GT-P5220|GT-I9200X|GT-N5110X|GT-N5120|SM-P905|SM-T111|SM-T2105|SM-T315|SM-T320|SM-T320X|SM-T321|SM-T520|SM-T525|SM-T530NU|SM-T230NU|SM-T330NU|SM-T900|XE500T1C|SM-P605V|SM-P905V|SM-T337V|SM-T537V|SM-T707V|SM-T807V|SM-P600X|SM-P900X|SM-T210X|SM-T230|SM-T230X|SM-T325|GT-P7503|SM-T531|SM-T330|SM-T530|SM-T705|SM-T705C|SM-T535|SM-T331|SM-T800|SM-T700|SM-T537|SM-T807|SM-P907A|SM-T337A|SM-T537A|SM-T707A|SM-T807A|SM-T237|SM-T807P|SM-P607T|SM-T217T|SM-T337T|SM-T807T|SM-T116NQ|SM-T116BU|SM-P550|SM-T350|SM-T550|SM-T9000|SM-P9000|SM-T705Y|SM-T805|GT-P3113|SM-T710|SM-T810|SM-T815|SM-T360|SM-T533|SM-T113|SM-T335|SM-T715|SM-T560|SM-T670|SM-T677|SM-T377|SM-T567|SM-T357T|SM-T555|SM-T561|SM-T713|SM-T719|SM-T813|SM-T819|SM-T580|SM-T355Y|SM-T280|SM-T817A|SM-T820|SM-W700|SM-P580|SM-T587|SM-P350|SM-P555M|SM-P355M|SM-T113NU|SM-T815Y', // SCH-P709|SCH-P729|SM-T2558|GT-I9205 - Samsung Mega - treat them like a regular phone.
  4375. // http://docs.aws.amazon.com/silk/latest/developerguide/user-agent.html
  4376. 'Kindle' => 'Kindle|Silk.*Accelerated|Android.*\b(KFOT|KFTT|KFJWI|KFJWA|KFOTE|KFSOWI|KFTHWI|KFTHWA|KFAPWI|KFAPWA|WFJWAE|KFSAWA|KFSAWI|KFASWI|KFARWI|KFFOWI|KFGIWI|KFMEWI)\b|Android.*Silk/[0-9.]+ like Chrome/[0-9.]+ (?!Mobile)',
  4377. // Only the Surface tablets with Windows RT are considered mobile.
  4378. // http://msdn.microsoft.com/en-us/library/ie/hh920767(v=vs.85).aspx
  4379. 'SurfaceTablet' => 'Windows NT [0-9.]+; ARM;.*(Tablet|ARMBJS)',
  4380. // http://shopping1.hp.com/is-bin/INTERSHOP.enfinity/WFS/WW-USSMBPublicStore-Site/en_US/-/USD/ViewStandardCatalog-Browse?CatalogCategoryID=JfIQ7EN5lqMAAAEyDcJUDwMT
  4381. 'HPTablet' => 'HP Slate (7|8|10)|HP ElitePad 900|hp-tablet|EliteBook.*Touch|HP 8|Slate 21|HP SlateBook 10',
  4382. // Watch out for PadFone, see #132.
  4383. // http://www.asus.com/de/Tablets_Mobile/Memo_Pad_Products/
  4384. 'AsusTablet' => '^.*PadFone((?!Mobile).)*$|Transformer|TF101|TF101G|TF300T|TF300TG|TF300TL|TF700T|TF700KL|TF701T|TF810C|ME171|ME301T|ME302C|ME371MG|ME370T|ME372MG|ME172V|ME173X|ME400C|Slider SL101|\bK00F\b|\bK00C\b|\bK00E\b|\bK00L\b|TX201LA|ME176C|ME102A|\bM80TA\b|ME372CL|ME560CG|ME372CG|ME302KL| K010 | K011 | K017 | K01E |ME572C|ME103K|ME170C|ME171C|\bME70C\b|ME581C|ME581CL|ME8510C|ME181C|P01Y|PO1MA|P01Z|\bP027\b',
  4385. 'BlackBerryTablet' => 'PlayBook|RIM Tablet',
  4386. 'HTCtablet' => 'HTC_Flyer_P512|HTC Flyer|HTC Jetstream|HTC-P715a|HTC EVO View 4G|PG41200|PG09410',
  4387. 'MotorolaTablet' => 'xoom|sholest|MZ615|MZ605|MZ505|MZ601|MZ602|MZ603|MZ604|MZ606|MZ607|MZ608|MZ609|MZ615|MZ616|MZ617',
  4388. 'NookTablet' => 'Android.*Nook|NookColor|nook browser|BNRV200|BNRV200A|BNTV250|BNTV250A|BNTV400|BNTV600|LogicPD Zoom2',
  4389. // http://www.acer.ro/ac/ro/RO/content/drivers
  4390. // http://www.packardbell.co.uk/pb/en/GB/content/download (Packard Bell is part of Acer)
  4391. // http://us.acer.com/ac/en/US/content/group/tablets
  4392. // http://www.acer.de/ac/de/DE/content/models/tablets/
  4393. // Can conflict with Micromax and Motorola phones codes.
  4394. 'AcerTablet' => 'Android.*; \b(A100|A101|A110|A200|A210|A211|A500|A501|A510|A511|A700|A701|W500|W500P|W501|W501P|W510|W511|W700|G100|G100W|B1-A71|B1-710|B1-711|A1-810|A1-811|A1-830)\b|W3-810|\bA3-A10\b|\bA3-A11\b|\bA3-A20\b|\bA3-A30',
  4395. // http://eu.computers.toshiba-europe.com/innovation/family/Tablets/1098744/banner_id/tablet_footerlink/
  4396. // http://us.toshiba.com/tablets/tablet-finder
  4397. // http://www.toshiba.co.jp/regza/tablet/
  4398. 'ToshibaTablet' => 'Android.*(AT100|AT105|AT200|AT205|AT270|AT275|AT300|AT305|AT1S5|AT500|AT570|AT700|AT830)|TOSHIBA.*FOLIO',
  4399. // http://www.nttdocomo.co.jp/english/service/developer/smart_phone/technical_info/spec/index.html
  4400. // http://www.lg.com/us/tablets
  4401. 'LGTablet' => '\bL-06C|LG-V909|LG-V900|LG-V700|LG-V510|LG-V500|LG-V410|LG-V400|LG-VK810\b',
  4402. 'FujitsuTablet' => 'Android.*\b(F-01D|F-02F|F-05E|F-10D|M532|Q572)\b',
  4403. // Prestigio Tablets http://www.prestigio.com/support
  4404. 'PrestigioTablet' => 'PMP3170B|PMP3270B|PMP3470B|PMP7170B|PMP3370B|PMP3570C|PMP5870C|PMP3670B|PMP5570C|PMP5770D|PMP3970B|PMP3870C|PMP5580C|PMP5880D|PMP5780D|PMP5588C|PMP7280C|PMP7280C3G|PMP7280|PMP7880D|PMP5597D|PMP5597|PMP7100D|PER3464|PER3274|PER3574|PER3884|PER5274|PER5474|PMP5097CPRO|PMP5097|PMP7380D|PMP5297C|PMP5297C_QUAD|PMP812E|PMP812E3G|PMP812F|PMP810E|PMP880TD|PMT3017|PMT3037|PMT3047|PMT3057|PMT7008|PMT5887|PMT5001|PMT5002',
  4405. // http://support.lenovo.com/en_GB/downloads/default.page?#
  4406. 'LenovoTablet' => 'Lenovo TAB|Idea(Tab|Pad)( A1|A10| K1|)|ThinkPad([ ]+)?Tablet|YT3-850M|YT3-X90L|YT3-X90F|YT3-X90X|Lenovo.*(S2109|S2110|S5000|S6000|K3011|A3000|A3500|A1000|A2107|A2109|A1107|A5500|A7600|B6000|B8000|B8080)(-|)(FL|F|HV|H|)',
  4407. // http://www.dell.com/support/home/us/en/04/Products/tab_mob/tablets
  4408. 'DellTablet' => 'Venue 11|Venue 8|Venue 7|Dell Streak 10|Dell Streak 7',
  4409. // http://www.yarvik.com/en/matrix/tablets/
  4410. 'YarvikTablet' => 'Android.*\b(TAB210|TAB211|TAB224|TAB250|TAB260|TAB264|TAB310|TAB360|TAB364|TAB410|TAB411|TAB420|TAB424|TAB450|TAB460|TAB461|TAB464|TAB465|TAB467|TAB468|TAB07-100|TAB07-101|TAB07-150|TAB07-151|TAB07-152|TAB07-200|TAB07-201-3G|TAB07-210|TAB07-211|TAB07-212|TAB07-214|TAB07-220|TAB07-400|TAB07-485|TAB08-150|TAB08-200|TAB08-201-3G|TAB08-201-30|TAB09-100|TAB09-211|TAB09-410|TAB10-150|TAB10-201|TAB10-211|TAB10-400|TAB10-410|TAB13-201|TAB274EUK|TAB275EUK|TAB374EUK|TAB462EUK|TAB474EUK|TAB9-200)\b',
  4411. 'MedionTablet' => 'Android.*\bOYO\b|LIFE.*(P9212|P9514|P9516|S9512)|LIFETAB',
  4412. 'ArnovaTablet' => '97G4|AN10G2|AN7bG3|AN7fG3|AN8G3|AN8cG3|AN7G3|AN9G3|AN7dG3|AN7dG3ST|AN7dG3ChildPad|AN10bG3|AN10bG3DT|AN9G2',
  4413. // http://www.intenso.de/kategorie_en.php?kategorie=33
  4414. // @todo: http://www.nbhkdz.com/read/b8e64202f92a2df129126bff.html - investigate
  4415. 'IntensoTablet' => 'INM8002KP|INM1010FP|INM805ND|Intenso Tab|TAB1004',
  4416. // IRU.ru Tablets http://www.iru.ru/catalog/soho/planetable/
  4417. 'IRUTablet' => 'M702pro',
  4418. 'MegafonTablet' => 'MegaFon V9|\bZTE V9\b|Android.*\bMT7A\b',
  4419. // http://www.e-boda.ro/tablete-pc.html
  4420. 'EbodaTablet' => 'E-Boda (Supreme|Impresspeed|Izzycomm|Essential)',
  4421. // http://www.allview.ro/produse/droseries/lista-tablete-pc/
  4422. 'AllViewTablet' => 'Allview.*(Viva|Alldro|City|Speed|All TV|Frenzy|Quasar|Shine|TX1|AX1|AX2)',
  4423. // http://wiki.archosfans.com/index.php?title=Main_Page
  4424. // @note Rewrite the regex format after we add more UAs.
  4425. 'ArchosTablet' => '\b(101G9|80G9|A101IT)\b|Qilive 97R|Archos5|\bARCHOS (70|79|80|90|97|101|FAMILYPAD|)(b|c|)(G10| Cobalt| TITANIUM(HD|)| Xenon| Neon|XSK| 2| XS 2| PLATINUM| CARBON|GAMEPAD)\b',
  4426. // http://www.ainol.com/plugin.php?identifier=ainol&module=product
  4427. 'AinolTablet' => 'NOVO7|NOVO8|NOVO10|Novo7Aurora|Novo7Basic|NOVO7PALADIN|novo9-Spark',
  4428. 'NokiaLumiaTablet' => 'Lumia 2520',
  4429. // @todo: inspect http://esupport.sony.com/US/p/select-system.pl?DIRECTOR=DRIVER
  4430. // Readers http://www.atsuhiro-me.net/ebook/sony-reader/sony-reader-web-browser
  4431. // http://www.sony.jp/support/tablet/
  4432. 'SonyTablet' => 'Sony.*Tablet|Xperia Tablet|Sony Tablet S|SO-03E|SGPT12|SGPT13|SGPT114|SGPT121|SGPT122|SGPT123|SGPT111|SGPT112|SGPT113|SGPT131|SGPT132|SGPT133|SGPT211|SGPT212|SGPT213|SGP311|SGP312|SGP321|EBRD1101|EBRD1102|EBRD1201|SGP351|SGP341|SGP511|SGP512|SGP521|SGP541|SGP551|SGP621|SGP612|SOT31',
  4433. // http://www.support.philips.com/support/catalog/worldproducts.jsp?userLanguage=en&userCountry=cn&categoryid=3G_LTE_TABLET_SU_CN_CARE&title=3G%20tablets%20/%20LTE%20range&_dyncharset=UTF-8
  4434. 'PhilipsTablet' => '\b(PI2010|PI3000|PI3100|PI3105|PI3110|PI3205|PI3210|PI3900|PI4010|PI7000|PI7100)\b',
  4435. // db + http://www.cube-tablet.com/buy-products.html
  4436. 'CubeTablet' => 'Android.*(K8GT|U9GT|U10GT|U16GT|U17GT|U18GT|U19GT|U20GT|U23GT|U30GT)|CUBE U8GT',
  4437. // http://www.cobyusa.com/?p=pcat&pcat_id=3001
  4438. 'CobyTablet' => 'MID1042|MID1045|MID1125|MID1126|MID7012|MID7014|MID7015|MID7034|MID7035|MID7036|MID7042|MID7048|MID7127|MID8042|MID8048|MID8127|MID9042|MID9740|MID9742|MID7022|MID7010',
  4439. // http://www.match.net.cn/products.asp
  4440. 'MIDTablet' => 'M9701|M9000|M9100|M806|M1052|M806|T703|MID701|MID713|MID710|MID727|MID760|MID830|MID728|MID933|MID125|MID810|MID732|MID120|MID930|MID800|MID731|MID900|MID100|MID820|MID735|MID980|MID130|MID833|MID737|MID960|MID135|MID860|MID736|MID140|MID930|MID835|MID733|MID4X10',
  4441. // http://www.msi.com/support
  4442. // @todo Research the Windows Tablets.
  4443. 'MSITablet' => 'MSI \b(Primo 73K|Primo 73L|Primo 81L|Primo 77|Primo 93|Primo 75|Primo 76|Primo 73|Primo 81|Primo 91|Primo 90|Enjoy 71|Enjoy 7|Enjoy 10)\b',
  4444. // @todo http://www.kyoceramobile.com/support/drivers/
  4445. // 'KyoceraTablet' => null,
  4446. // @todo http://intexuae.com/index.php/category/mobile-devices/tablets-products/
  4447. // 'IntextTablet' => null,
  4448. // http://pdadb.net/index.php?m=pdalist&list=SMiT (NoName Chinese Tablets)
  4449. // http://www.imp3.net/14/show.php?itemid=20454
  4450. 'SMiTTablet' => 'Android.*(\bMID\b|MID-560|MTV-T1200|MTV-PND531|MTV-P1101|MTV-PND530)',
  4451. // http://www.rock-chips.com/index.php?do=prod&pid=2
  4452. 'RockChipTablet' => 'Android.*(RK2818|RK2808A|RK2918|RK3066)|RK2738|RK2808A',
  4453. // http://www.fly-phone.com/devices/tablets/ ; http://www.fly-phone.com/service/
  4454. 'FlyTablet' => 'IQ310|Fly Vision',
  4455. // http://www.bqreaders.com/gb/tablets-prices-sale.html
  4456. 'bqTablet' => 'Android.*(bq)?.*(Elcano|Curie|Edison|Maxwell|Kepler|Pascal|Tesla|Hypatia|Platon|Newton|Livingstone|Cervantes|Avant|Aquaris [E|M]10)|Maxwell.*Lite|Maxwell.*Plus',
  4457. // http://www.huaweidevice.com/worldwide/productFamily.do?method=index&directoryId=5011&treeId=3290
  4458. // http://www.huaweidevice.com/worldwide/downloadCenter.do?method=index&directoryId=3372&treeId=0&tb=1&type=software (including legacy tablets)
  4459. 'HuaweiTablet' => 'MediaPad|MediaPad 7 Youth|IDEOS S7|S7-201c|S7-202u|S7-101|S7-103|S7-104|S7-105|S7-106|S7-201|S7-Slim',
  4460. // Nec or Medias Tab
  4461. 'NecTablet' => '\bN-06D|\bN-08D',
  4462. // Pantech Tablets: http://www.pantechusa.com/phones/
  4463. 'PantechTablet' => 'Pantech.*P4100',
  4464. // Broncho Tablets: http://www.broncho.cn/ (hard to find)
  4465. 'BronchoTablet' => 'Broncho.*(N701|N708|N802|a710)',
  4466. // http://versusuk.com/support.html
  4467. 'VersusTablet' => 'TOUCHPAD.*[78910]|\bTOUCHTAB\b',
  4468. // http://www.zync.in/index.php/our-products/tablet-phablets
  4469. 'ZyncTablet' => 'z1000|Z99 2G|z99|z930|z999|z990|z909|Z919|z900',
  4470. // http://www.positivoinformatica.com.br/www/pessoal/tablet-ypy/
  4471. 'PositivoTablet' => 'TB07STA|TB10STA|TB07FTA|TB10FTA',
  4472. // https://www.nabitablet.com/
  4473. 'NabiTablet' => 'Android.*\bNabi',
  4474. 'KoboTablet' => 'Kobo Touch|\bK080\b|\bVox\b Build|\bArc\b Build',
  4475. // French Danew Tablets http://www.danew.com/produits-tablette.php
  4476. 'DanewTablet' => 'DSlide.*\b(700|701R|702|703R|704|802|970|971|972|973|974|1010|1012)\b',
  4477. // Texet Tablets and Readers http://www.texet.ru/tablet/
  4478. 'TexetTablet' => 'NaviPad|TB-772A|TM-7045|TM-7055|TM-9750|TM-7016|TM-7024|TM-7026|TM-7041|TM-7043|TM-7047|TM-8041|TM-9741|TM-9747|TM-9748|TM-9751|TM-7022|TM-7021|TM-7020|TM-7011|TM-7010|TM-7023|TM-7025|TM-7037W|TM-7038W|TM-7027W|TM-9720|TM-9725|TM-9737W|TM-1020|TM-9738W|TM-9740|TM-9743W|TB-807A|TB-771A|TB-727A|TB-725A|TB-719A|TB-823A|TB-805A|TB-723A|TB-715A|TB-707A|TB-705A|TB-709A|TB-711A|TB-890HD|TB-880HD|TB-790HD|TB-780HD|TB-770HD|TB-721HD|TB-710HD|TB-434HD|TB-860HD|TB-840HD|TB-760HD|TB-750HD|TB-740HD|TB-730HD|TB-722HD|TB-720HD|TB-700HD|TB-500HD|TB-470HD|TB-431HD|TB-430HD|TB-506|TB-504|TB-446|TB-436|TB-416|TB-146SE|TB-126SE',
  4479. // Avoid detecting 'PLAYSTATION 3' as mobile.
  4480. 'PlaystationTablet' => 'Playstation.*(Portable|Vita)',
  4481. // http://www.trekstor.de/surftabs.html
  4482. 'TrekstorTablet' => 'ST10416-1|VT10416-1|ST70408-1|ST702xx-1|ST702xx-2|ST80208|ST97216|ST70104-2|VT10416-2|ST10216-2A|SurfTab',
  4483. // http://www.pyleaudio.com/Products.aspx?%2fproducts%2fPersonal-Electronics%2fTablets
  4484. 'PyleAudioTablet' => '\b(PTBL10CEU|PTBL10C|PTBL72BC|PTBL72BCEU|PTBL7CEU|PTBL7C|PTBL92BC|PTBL92BCEU|PTBL9CEU|PTBL9CUK|PTBL9C)\b',
  4485. // http://www.advandigital.com/index.php?link=content-product&jns=JP001
  4486. // because of the short codenames we have to include whitespaces to reduce the possible conflicts.
  4487. 'AdvanTablet' => 'Android.* \b(E3A|T3X|T5C|T5B|T3E|T3C|T3B|T1J|T1F|T2A|T1H|T1i|E1C|T1-E|T5-A|T4|E1-B|T2Ci|T1-B|T1-D|O1-A|E1-A|T1-A|T3A|T4i)\b ',
  4488. // http://www.danytech.com/category/tablet-pc
  4489. 'DanyTechTablet' => 'Genius Tab G3|Genius Tab S2|Genius Tab Q3|Genius Tab G4|Genius Tab Q4|Genius Tab G-II|Genius TAB GII|Genius TAB GIII|Genius Tab S1',
  4490. // http://www.galapad.net/product.html
  4491. 'GalapadTablet' => 'Android.*\bG1\b',
  4492. // http://www.micromaxinfo.com/tablet/funbook
  4493. 'MicromaxTablet' => 'Funbook|Micromax.*\b(P250|P560|P360|P362|P600|P300|P350|P500|P275)\b',
  4494. // http://www.karbonnmobiles.com/products_tablet.php
  4495. 'KarbonnTablet' => 'Android.*\b(A39|A37|A34|ST8|ST10|ST7|Smart Tab3|Smart Tab2)\b',
  4496. // http://www.myallfine.com/Products.asp
  4497. 'AllFineTablet' => 'Fine7 Genius|Fine7 Shine|Fine7 Air|Fine8 Style|Fine9 More|Fine10 Joy|Fine11 Wide',
  4498. // http://www.proscanvideo.com/products-search.asp?itemClass=TABLET&itemnmbr=
  4499. 'PROSCANTablet' => '\b(PEM63|PLT1023G|PLT1041|PLT1044|PLT1044G|PLT1091|PLT4311|PLT4311PL|PLT4315|PLT7030|PLT7033|PLT7033D|PLT7035|PLT7035D|PLT7044K|PLT7045K|PLT7045KB|PLT7071KG|PLT7072|PLT7223G|PLT7225G|PLT7777G|PLT7810K|PLT7849G|PLT7851G|PLT7852G|PLT8015|PLT8031|PLT8034|PLT8036|PLT8080K|PLT8082|PLT8088|PLT8223G|PLT8234G|PLT8235G|PLT8816K|PLT9011|PLT9045K|PLT9233G|PLT9735|PLT9760G|PLT9770G)\b',
  4500. // http://www.yonesnav.com/products/products.php
  4501. 'YONESTablet' => 'BQ1078|BC1003|BC1077|RK9702|BC9730|BC9001|IT9001|BC7008|BC7010|BC708|BC728|BC7012|BC7030|BC7027|BC7026',
  4502. // http://www.cjshowroom.com/eproducts.aspx?classcode=004001001
  4503. // China manufacturer makes tablets for different small brands (eg. http://www.zeepad.net/index.html)
  4504. 'ChangJiaTablet' => 'TPC7102|TPC7103|TPC7105|TPC7106|TPC7107|TPC7201|TPC7203|TPC7205|TPC7210|TPC7708|TPC7709|TPC7712|TPC7110|TPC8101|TPC8103|TPC8105|TPC8106|TPC8203|TPC8205|TPC8503|TPC9106|TPC9701|TPC97101|TPC97103|TPC97105|TPC97106|TPC97111|TPC97113|TPC97203|TPC97603|TPC97809|TPC97205|TPC10101|TPC10103|TPC10106|TPC10111|TPC10203|TPC10205|TPC10503',
  4505. // http://www.gloryunion.cn/products.asp
  4506. // http://www.allwinnertech.com/en/apply/mobile.html
  4507. // http://www.ptcl.com.pk/pd_content.php?pd_id=284 (EVOTAB)
  4508. // @todo: Softwiner tablets?
  4509. // aka. Cute or Cool tablets. Not sure yet, must research to avoid collisions.
  4510. 'GUTablet' => 'TX-A1301|TX-M9002|Q702|kf026', // A12R|D75A|D77|D79|R83|A95|A106C|R15|A75|A76|D71|D72|R71|R73|R77|D82|R85|D92|A97|D92|R91|A10F|A77F|W71F|A78F|W78F|W81F|A97F|W91F|W97F|R16G|C72|C73E|K72|K73|R96G
  4511. // http://www.pointofview-online.com/showroom.php?shop_mode=product_listing&category_id=118
  4512. 'PointOfViewTablet' => 'TAB-P506|TAB-navi-7-3G-M|TAB-P517|TAB-P-527|TAB-P701|TAB-P703|TAB-P721|TAB-P731N|TAB-P741|TAB-P825|TAB-P905|TAB-P925|TAB-PR945|TAB-PL1015|TAB-P1025|TAB-PI1045|TAB-P1325|TAB-PROTAB[0-9]+|TAB-PROTAB25|TAB-PROTAB26|TAB-PROTAB27|TAB-PROTAB26XL|TAB-PROTAB2-IPS9|TAB-PROTAB30-IPS9|TAB-PROTAB25XXL|TAB-PROTAB26-IPS10|TAB-PROTAB30-IPS10',
  4513. // http://www.overmax.pl/pl/katalog-produktow,p8/tablety,c14/
  4514. // @todo: add more tests.
  4515. 'OvermaxTablet' => 'OV-(SteelCore|NewBase|Basecore|Baseone|Exellen|Quattor|EduTab|Solution|ACTION|BasicTab|TeddyTab|MagicTab|Stream|TB-08|TB-09)',
  4516. // http://hclmetablet.com/India/index.php
  4517. 'HCLTablet' => 'HCL.*Tablet|Connect-3G-2.0|Connect-2G-2.0|ME Tablet U1|ME Tablet U2|ME Tablet G1|ME Tablet X1|ME Tablet Y2|ME Tablet Sync',
  4518. // http://www.edigital.hu/Tablet_es_e-book_olvaso/Tablet-c18385.html
  4519. 'DPSTablet' => 'DPS Dream 9|DPS Dual 7',
  4520. // http://www.visture.com/index.asp
  4521. 'VistureTablet' => 'V97 HD|i75 3G|Visture V4( HD)?|Visture V5( HD)?|Visture V10',
  4522. // http://www.mijncresta.nl/tablet
  4523. 'CrestaTablet' => 'CTP(-)?810|CTP(-)?818|CTP(-)?828|CTP(-)?838|CTP(-)?888|CTP(-)?978|CTP(-)?980|CTP(-)?987|CTP(-)?988|CTP(-)?989',
  4524. // MediaTek - http://www.mediatek.com/_en/01_products/02_proSys.php?cata_sn=1&cata1_sn=1&cata2_sn=309
  4525. 'MediatekTablet' => '\bMT8125|MT8389|MT8135|MT8377\b',
  4526. // Concorde tab
  4527. 'ConcordeTablet' => 'Concorde([ ]+)?Tab|ConCorde ReadMan',
  4528. // GoClever Tablets - http://www.goclever.com/uk/products,c1/tablet,c5/
  4529. 'GoCleverTablet' => 'GOCLEVER TAB|A7GOCLEVER|M1042|M7841|M742|R1042BK|R1041|TAB A975|TAB A7842|TAB A741|TAB A741L|TAB M723G|TAB M721|TAB A1021|TAB I921|TAB R721|TAB I720|TAB T76|TAB R70|TAB R76.2|TAB R106|TAB R83.2|TAB M813G|TAB I721|GCTA722|TAB I70|TAB I71|TAB S73|TAB R73|TAB R74|TAB R93|TAB R75|TAB R76.1|TAB A73|TAB A93|TAB A93.2|TAB T72|TAB R83|TAB R974|TAB R973|TAB A101|TAB A103|TAB A104|TAB A104.2|R105BK|M713G|A972BK|TAB A971|TAB R974.2|TAB R104|TAB R83.3|TAB A1042',
  4530. // Modecom Tablets - http://www.modecom.eu/tablets/portal/
  4531. 'ModecomTablet' => 'FreeTAB 9000|FreeTAB 7.4|FreeTAB 7004|FreeTAB 7800|FreeTAB 2096|FreeTAB 7.5|FreeTAB 1014|FreeTAB 1001 |FreeTAB 8001|FreeTAB 9706|FreeTAB 9702|FreeTAB 7003|FreeTAB 7002|FreeTAB 1002|FreeTAB 7801|FreeTAB 1331|FreeTAB 1004|FreeTAB 8002|FreeTAB 8014|FreeTAB 9704|FreeTAB 1003',
  4532. // Vonino Tablets - http://www.vonino.eu/tablets
  4533. 'VoninoTablet' => '\b(Argus[ _]?S|Diamond[ _]?79HD|Emerald[ _]?78E|Luna[ _]?70C|Onyx[ _]?S|Onyx[ _]?Z|Orin[ _]?HD|Orin[ _]?S|Otis[ _]?S|SpeedStar[ _]?S|Magnet[ _]?M9|Primus[ _]?94[ _]?3G|Primus[ _]?94HD|Primus[ _]?QS|Android.*\bQ8\b|Sirius[ _]?EVO[ _]?QS|Sirius[ _]?QS|Spirit[ _]?S)\b',
  4534. // ECS Tablets - http://www.ecs.com.tw/ECSWebSite/Product/Product_Tablet_List.aspx?CategoryID=14&MenuID=107&childid=M_107&LanID=0
  4535. 'ECSTablet' => 'V07OT2|TM105A|S10OT1|TR10CS1',
  4536. // Storex Tablets - http://storex.fr/espace_client/support.html
  4537. // @note: no need to add all the tablet codes since they are guided by the first regex.
  4538. 'StorexTablet' => 'eZee[_\']?(Tab|Go)[0-9]+|TabLC7|Looney Tunes Tab',
  4539. // Generic Vodafone tablets.
  4540. 'VodafoneTablet' => 'SmartTab([ ]+)?[0-9]+|SmartTabII10|SmartTabII7|VF-1497',
  4541. // French tablets - Essentiel B http://www.boulanger.fr/tablette_tactile_e-book/tablette_tactile_essentiel_b/cl_68908.htm?multiChoiceToDelete=brand&mc_brand=essentielb
  4542. // Aka: http://www.essentielb.fr/
  4543. 'EssentielBTablet' => 'Smart[ \']?TAB[ ]+?[0-9]+|Family[ \']?TAB2',
  4544. // Ross & Moor - http://ross-moor.ru/
  4545. 'RossMoorTablet' => 'RM-790|RM-997|RMD-878G|RMD-974R|RMT-705A|RMT-701|RME-601|RMT-501|RMT-711',
  4546. // i-mobile http://product.i-mobilephone.com/Mobile_Device
  4547. 'iMobileTablet' => 'i-mobile i-note',
  4548. // http://www.tolino.de/de/vergleichen/
  4549. 'TolinoTablet' => 'tolino tab [0-9.]+|tolino shine',
  4550. // AudioSonic - a Kmart brand
  4551. // http://www.kmart.com.au/webapp/wcs/stores/servlet/Search?langId=-1&storeId=10701&catalogId=10001&categoryId=193001&pageSize=72&currentPage=1&searchCategory=193001%2b4294965664&sortBy=p_MaxPrice%7c1
  4552. 'AudioSonicTablet' => '\bC-22Q|T7-QC|T-17B|T-17P\b',
  4553. // AMPE Tablets - http://www.ampe.com.my/product-category/tablets/
  4554. // @todo: add them gradually to avoid conflicts.
  4555. 'AMPETablet' => 'Android.* A78 ',
  4556. // Skk Mobile - http://skkmobile.com.ph/product_tablets.php
  4557. 'SkkTablet' => 'Android.* (SKYPAD|PHOENIX|CYCLOPS)',
  4558. // Tecno Mobile (only tablet) - http://www.tecno-mobile.com/index.php/product?filterby=smart&list_order=all&page=1
  4559. 'TecnoTablet' => 'TECNO P9',
  4560. // JXD (consoles & tablets) - http://jxd.hk/products.asp?selectclassid=009008&clsid=3
  4561. 'JXDTablet' => 'Android.* \b(F3000|A3300|JXD5000|JXD3000|JXD2000|JXD300B|JXD300|S5800|S7800|S602b|S5110b|S7300|S5300|S602|S603|S5100|S5110|S601|S7100a|P3000F|P3000s|P101|P200s|P1000m|P200m|P9100|P1000s|S6600b|S908|P1000|P300|S18|S6600|S9100)\b',
  4562. // i-Joy tablets - http://www.i-joy.es/en/cat/products/tablets/
  4563. 'iJoyTablet' => 'Tablet (Spirit 7|Essentia|Galatea|Fusion|Onix 7|Landa|Titan|Scooby|Deox|Stella|Themis|Argon|Unique 7|Sygnus|Hexen|Finity 7|Cream|Cream X2|Jade|Neon 7|Neron 7|Kandy|Scape|Saphyr 7|Rebel|Biox|Rebel|Rebel 8GB|Myst|Draco 7|Myst|Tab7-004|Myst|Tadeo Jones|Tablet Boing|Arrow|Draco Dual Cam|Aurix|Mint|Amity|Revolution|Finity 9|Neon 9|T9w|Amity 4GB Dual Cam|Stone 4GB|Stone 8GB|Andromeda|Silken|X2|Andromeda II|Halley|Flame|Saphyr 9,7|Touch 8|Planet|Triton|Unique 10|Hexen 10|Memphis 4GB|Memphis 8GB|Onix 10)',
  4564. // http://www.intracon.eu/tablet
  4565. 'FX2Tablet' => 'FX2 PAD7|FX2 PAD10',
  4566. // http://www.xoro.de/produkte/
  4567. // @note: Might be the same brand with 'Simply tablets'
  4568. 'XoroTablet' => 'KidsPAD 701|PAD[ ]?712|PAD[ ]?714|PAD[ ]?716|PAD[ ]?717|PAD[ ]?718|PAD[ ]?720|PAD[ ]?721|PAD[ ]?722|PAD[ ]?790|PAD[ ]?792|PAD[ ]?900|PAD[ ]?9715D|PAD[ ]?9716DR|PAD[ ]?9718DR|PAD[ ]?9719QR|PAD[ ]?9720QR|TelePAD1030|Telepad1032|TelePAD730|TelePAD731|TelePAD732|TelePAD735Q|TelePAD830|TelePAD9730|TelePAD795|MegaPAD 1331|MegaPAD 1851|MegaPAD 2151',
  4569. // http://www1.viewsonic.com/products/computing/tablets/
  4570. 'ViewsonicTablet' => 'ViewPad 10pi|ViewPad 10e|ViewPad 10s|ViewPad E72|ViewPad7|ViewPad E100|ViewPad 7e|ViewSonic VB733|VB100a',
  4571. // http://www.odys.de/web/internet-tablet_en.html
  4572. 'OdysTablet' => 'LOOX|XENO10|ODYS[ -](Space|EVO|Xpress|NOON)|\bXELIO\b|Xelio10Pro|XELIO7PHONETAB|XELIO10EXTREME|XELIOPT2|NEO_QUAD10',
  4573. // http://www.captiva-power.de/products.html#tablets-en
  4574. 'CaptivaTablet' => 'CAPTIVA PAD',
  4575. // IconBIT - http://www.iconbit.com/products/tablets/
  4576. 'IconbitTablet' => 'NetTAB|NT-3702|NT-3702S|NT-3702S|NT-3603P|NT-3603P|NT-0704S|NT-0704S|NT-3805C|NT-3805C|NT-0806C|NT-0806C|NT-0909T|NT-0909T|NT-0907S|NT-0907S|NT-0902S|NT-0902S',
  4577. // http://www.teclast.com/topic.php?channelID=70&topicID=140&pid=63
  4578. 'TeclastTablet' => 'T98 4G|\bP80\b|\bX90HD\b|X98 Air|X98 Air 3G|\bX89\b|P80 3G|\bX80h\b|P98 Air|\bX89HD\b|P98 3G|\bP90HD\b|P89 3G|X98 3G|\bP70h\b|P79HD 3G|G18d 3G|\bP79HD\b|\bP89s\b|\bA88\b|\bP10HD\b|\bP19HD\b|G18 3G|\bP78HD\b|\bA78\b|\bP75\b|G17s 3G|G17h 3G|\bP85t\b|\bP90\b|\bP11\b|\bP98t\b|\bP98HD\b|\bG18d\b|\bP85s\b|\bP11HD\b|\bP88s\b|\bA80HD\b|\bA80se\b|\bA10h\b|\bP89\b|\bP78s\b|\bG18\b|\bP85\b|\bA70h\b|\bA70\b|\bG17\b|\bP18\b|\bA80s\b|\bA11s\b|\bP88HD\b|\bA80h\b|\bP76s\b|\bP76h\b|\bP98\b|\bA10HD\b|\bP78\b|\bP88\b|\bA11\b|\bA10t\b|\bP76a\b|\bP76t\b|\bP76e\b|\bP85HD\b|\bP85a\b|\bP86\b|\bP75HD\b|\bP76v\b|\bA12\b|\bP75a\b|\bA15\b|\bP76Ti\b|\bP81HD\b|\bA10\b|\bT760VE\b|\bT720HD\b|\bP76\b|\bP73\b|\bP71\b|\bP72\b|\bT720SE\b|\bC520Ti\b|\bT760\b|\bT720VE\b|T720-3GE|T720-WiFi',
  4579. // Onda - http://www.onda-tablet.com/buy-android-onda.html?dir=desc&limit=all&order=price
  4580. 'OndaTablet' => '\b(V975i|Vi30|VX530|V701|Vi60|V701s|Vi50|V801s|V719|Vx610w|VX610W|V819i|Vi10|VX580W|Vi10|V711s|V813|V811|V820w|V820|Vi20|V711|VI30W|V712|V891w|V972|V819w|V820w|Vi60|V820w|V711|V813s|V801|V819|V975s|V801|V819|V819|V818|V811|V712|V975m|V101w|V961w|V812|V818|V971|V971s|V919|V989|V116w|V102w|V973|Vi40)\b[\s]+',
  4581. 'JaytechTablet' => 'TPC-PA762',
  4582. 'BlaupunktTablet' => 'Endeavour 800NG|Endeavour 1010',
  4583. // http://www.digma.ru/support/download/
  4584. // @todo: Ebooks also (if requested)
  4585. 'DigmaTablet' => '\b(iDx10|iDx9|iDx8|iDx7|iDxD7|iDxD8|iDsQ8|iDsQ7|iDsQ8|iDsD10|iDnD7|3TS804H|iDsQ11|iDj7|iDs10)\b',
  4586. // http://www.evolioshop.com/ro/tablete-pc.html
  4587. // http://www.evolio.ro/support/downloads_static.html?cat=2
  4588. // @todo: Research some more
  4589. 'EvolioTablet' => 'ARIA_Mini_wifi|Aria[ _]Mini|Evolio X10|Evolio X7|Evolio X8|\bEvotab\b|\bNeura\b',
  4590. // @todo http://www.lavamobiles.com/tablets-data-cards
  4591. 'LavaTablet' => 'QPAD E704|\bIvoryS\b|E-TAB IVORY|\bE-TAB\b',
  4592. // http://www.breezetablet.com/
  4593. 'AocTablet' => 'MW0811|MW0812|MW0922|MTK8382|MW1031|MW0831|MW0821|MW0931|MW0712',
  4594. // http://www.mpmaneurope.com/en/products/internet-tablets-14/android-tablets-14/
  4595. 'MpmanTablet' => 'MP11 OCTA|MP10 OCTA|MPQC1114|MPQC1004|MPQC994|MPQC974|MPQC973|MPQC804|MPQC784|MPQC780|\bMPG7\b|MPDCG75|MPDCG71|MPDC1006|MP101DC|MPDC9000|MPDC905|MPDC706HD|MPDC706|MPDC705|MPDC110|MPDC100|MPDC99|MPDC97|MPDC88|MPDC8|MPDC77|MP709|MID701|MID711|MID170|MPDC703|MPQC1010',
  4596. // https://www.celkonmobiles.com/?_a=categoryphones&sid=2
  4597. 'CelkonTablet' => 'CT695|CT888|CT[\s]?910|CT7 Tab|CT9 Tab|CT3 Tab|CT2 Tab|CT1 Tab|C820|C720|\bCT-1\b',
  4598. // http://www.wolderelectronics.com/productos/manuales-y-guias-rapidas/categoria-2-miTab
  4599. 'WolderTablet' => 'miTab \b(DIAMOND|SPACE|BROOKLYN|NEO|FLY|MANHATTAN|FUNK|EVOLUTION|SKY|GOCAR|IRON|GENIUS|POP|MINT|EPSILON|BROADWAY|JUMP|HOP|LEGEND|NEW AGE|LINE|ADVANCE|FEEL|FOLLOW|LIKE|LINK|LIVE|THINK|FREEDOM|CHICAGO|CLEVELAND|BALTIMORE-GH|IOWA|BOSTON|SEATTLE|PHOENIX|DALLAS|IN 101|MasterChef)\b',
  4600. // http://www.mi.com/en
  4601. 'MiTablet' => '\bMI PAD\b|\bHM NOTE 1W\b',
  4602. // http://www.nbru.cn/index.html
  4603. 'NibiruTablet' => 'Nibiru M1|Nibiru Jupiter One',
  4604. // http://navroad.com/products/produkty/tablety/
  4605. // http://navroad.com/products/produkty/tablety/
  4606. 'NexoTablet' => 'NEXO NOVA|NEXO 10|NEXO AVIO|NEXO FREE|NEXO GO|NEXO EVO|NEXO 3G|NEXO SMART|NEXO KIDDO|NEXO MOBI',
  4607. // http://leader-online.com/new_site/product-category/tablets/
  4608. // http://www.leader-online.net.au/List/Tablet
  4609. 'LeaderTablet' => 'TBLT10Q|TBLT10I|TBL-10WDKB|TBL-10WDKBO2013|TBL-W230V2|TBL-W450|TBL-W500|SV572|TBLT7I|TBA-AC7-8G|TBLT79|TBL-8W16|TBL-10W32|TBL-10WKB|TBL-W100',
  4610. // http://www.datawind.com/ubislate/
  4611. 'UbislateTablet' => 'UbiSlate[\s]?7C',
  4612. // http://www.pocketbook-int.com/ru/support
  4613. 'PocketBookTablet' => 'Pocketbook',
  4614. // http://www.kocaso.com/product_tablet.html
  4615. 'KocasoTablet' => '\b(TB-1207)\b',
  4616. // http://global.hisense.com/product/asia/tablet/Sero7/201412/t20141215_91832.htm
  4617. 'HisenseTablet' => '\b(F5281|E2371)\b',
  4618. // http://www.tesco.com/direct/hudl/
  4619. 'Hudl' => 'Hudl HT7S3|Hudl 2',
  4620. // http://www.telstra.com.au/home-phone/thub-2/
  4621. 'TelstraTablet' => 'T-Hub2',
  4622. 'GenericTablet' => 'Android.*\b97D\b|Tablet(?!.*PC)|BNTV250A|MID-WCDMA|LogicPD Zoom2|\bA7EB\b|CatNova8|A1_07|CT704|CT1002|\bM721\b|rk30sdk|\bEVOTAB\b|M758A|ET904|ALUMIUM10|Smartfren Tab|Endeavour 1010|Tablet-PC-4|Tagi Tab|\bM6pro\b|CT1020W|arc 10HD|\bTP750\b|\bQTAQZ3\b'
  4623. );
  4624. /**
  4625. * List of mobile Operating Systems.
  4626. *
  4627. * @var array
  4628. */
  4629. protected static $operatingSystems = array(
  4630. 'AndroidOS' => 'Android',
  4631. 'BlackBerryOS' => 'blackberry|\bBB10\b|rim tablet os',
  4632. 'PalmOS' => 'PalmOS|avantgo|blazer|elaine|hiptop|palm|plucker|xiino',
  4633. 'SymbianOS' => 'Symbian|SymbOS|Series60|Series40|SYB-[0-9]+|\bS60\b',
  4634. // @reference: http://en.wikipedia.org/wiki/Windows_Mobile
  4635. 'WindowsMobileOS' => 'Windows CE.*(PPC|Smartphone|Mobile|[0-9]{3}x[0-9]{3})|Window Mobile|Windows Phone [0-9.]+|WCE;',
  4636. // @reference: http://en.wikipedia.org/wiki/Windows_Phone
  4637. // http://wifeng.cn/?r=blog&a=view&id=106
  4638. // http://nicksnettravels.builttoroam.com/post/2011/01/10/Bogus-Windows-Phone-7-User-Agent-String.aspx
  4639. // http://msdn.microsoft.com/library/ms537503.aspx
  4640. // https://msdn.microsoft.com/en-us/library/hh869301(v=vs.85).aspx
  4641. 'WindowsPhoneOS' => 'Windows Phone 10.0|Windows Phone 8.1|Windows Phone 8.0|Windows Phone OS|XBLWP7|ZuneWP7|Windows NT 6.[23]; ARM;',
  4642. 'iOS' => '\biPhone.*Mobile|\biPod|\biPad',
  4643. // http://en.wikipedia.org/wiki/MeeGo
  4644. // @todo: research MeeGo in UAs
  4645. 'MeeGoOS' => 'MeeGo',
  4646. // http://en.wikipedia.org/wiki/Maemo
  4647. // @todo: research Maemo in UAs
  4648. 'MaemoOS' => 'Maemo',
  4649. 'JavaOS' => 'J2ME/|\bMIDP\b|\bCLDC\b', // '|Java/' produces bug #135
  4650. 'webOS' => 'webOS|hpwOS',
  4651. 'badaOS' => '\bBada\b',
  4652. 'BREWOS' => 'BREW',
  4653. );
  4654. /**
  4655. * List of mobile User Agents.
  4656. *
  4657. * IMPORTANT: This is a list of only mobile browsers.
  4658. * Mobile Detect 2.x supports only mobile browsers,
  4659. * it was never designed to detect all browsers.
  4660. * The change will come in 2017 in the 3.x release for PHP7.
  4661. *
  4662. * @var array
  4663. */
  4664. protected static $browsers = array(
  4665. //'Vivaldi' => 'Vivaldi',
  4666. // @reference: https://developers.google.com/chrome/mobile/docs/user-agent
  4667. 'Chrome' => '\bCrMo\b|CriOS|Android.*Chrome/[.0-9]* (Mobile)?',
  4668. 'Dolfin' => '\bDolfin\b',
  4669. 'Opera' => 'Opera.*Mini|Opera.*Mobi|Android.*Opera|Mobile.*OPR/[0-9.]+|Coast/[0-9.]+',
  4670. 'Skyfire' => 'Skyfire',
  4671. 'Edge' => 'Mobile Safari/[.0-9]* Edge',
  4672. 'IE' => 'IEMobile|MSIEMobile', // |Trident/[.0-9]+
  4673. 'Firefox' => 'fennec|firefox.*maemo|(Mobile|Tablet).*Firefox|Firefox.*Mobile|FxiOS',
  4674. 'Bolt' => 'bolt',
  4675. 'TeaShark' => 'teashark',
  4676. 'Blazer' => 'Blazer',
  4677. // @reference: http://developer.apple.com/library/safari/#documentation/AppleApplications/Reference/SafariWebContent/OptimizingforSafarioniPhone/OptimizingforSafarioniPhone.html#//apple_ref/doc/uid/TP40006517-SW3
  4678. 'Safari' => 'Version.*Mobile.*Safari|Safari.*Mobile|MobileSafari',
  4679. // http://en.wikipedia.org/wiki/Midori_(web_browser)
  4680. //'Midori' => 'midori',
  4681. //'Tizen' => 'Tizen',
  4682. 'UCBrowser' => 'UC.*Browser|UCWEB',
  4683. 'baiduboxapp' => 'baiduboxapp',
  4684. 'baidubrowser' => 'baidubrowser',
  4685. // https://github.com/serbanghita/Mobile-Detect/issues/7
  4686. 'DiigoBrowser' => 'DiigoBrowser',
  4687. // http://www.puffinbrowser.com/index.php
  4688. 'Puffin' => 'Puffin',
  4689. // http://mercury-browser.com/index.html
  4690. 'Mercury' => '\bMercury\b',
  4691. // http://en.wikipedia.org/wiki/Obigo_Browser
  4692. 'ObigoBrowser' => 'Obigo',
  4693. // http://en.wikipedia.org/wiki/NetFront
  4694. 'NetFront' => 'NF-Browser',
  4695. // @reference: http://en.wikipedia.org/wiki/Minimo
  4696. // http://en.wikipedia.org/wiki/Vision_Mobile_Browser
  4697. 'GenericBrowser' => 'NokiaBrowser|OviBrowser|OneBrowser|TwonkyBeamBrowser|SEMC.*Browser|FlyFlow|Minimo|NetFront|Novarra-Vision|MQQBrowser|MicroMessenger',
  4698. // @reference: https://en.wikipedia.org/wiki/Pale_Moon_(web_browser)
  4699. 'PaleMoon' => 'Android.*PaleMoon|Mobile.*PaleMoon',
  4700. );
  4701. /**
  4702. * Utilities.
  4703. *
  4704. * @var array
  4705. */
  4706. protected static $utilities = array(
  4707. // Experimental. When a mobile device wants to switch to 'Desktop Mode'.
  4708. // http://scottcate.com/technology/windows-phone-8-ie10-desktop-or-mobile/
  4709. // https://github.com/serbanghita/Mobile-Detect/issues/57#issuecomment-15024011
  4710. // https://developers.facebook.com/docs/sharing/best-practices
  4711. 'Bot' => 'Googlebot|facebookexternalhit|AdsBot-Google|Google Keyword Suggestion|Facebot|YandexBot|YandexMobileBot|bingbot|ia_archiver|AhrefsBot|Ezooms|GSLFbot|WBSearchBot|Twitterbot|TweetmemeBot|Twikle|PaperLiBot|Wotbox|UnwindFetchor|Exabot|MJ12bot|YandexImages|TurnitinBot|Pingdom',
  4712. 'MobileBot' => 'Googlebot-Mobile|AdsBot-Google-Mobile|YahooSeeker/M1A1-R2D2',
  4713. 'DesktopMode' => 'WPDesktop',
  4714. 'TV' => 'SonyDTV|HbbTV', // experimental
  4715. 'WebKit' => '(webkit)[ /]([\w.]+)',
  4716. // @todo: Include JXD consoles.
  4717. 'Console' => '\b(Nintendo|Nintendo WiiU|Nintendo 3DS|PLAYSTATION|Xbox)\b',
  4718. 'Watch' => 'SM-V700',
  4719. );
  4720. /**
  4721. * All possible HTTP headers that represent the
  4722. * User-Agent string.
  4723. *
  4724. * @var array
  4725. */
  4726. protected static $uaHttpHeaders = array(
  4727. // The default User-Agent string.
  4728. 'HTTP_USER_AGENT',
  4729. // Header can occur on devices using Opera Mini.
  4730. 'HTTP_X_OPERAMINI_PHONE_UA',
  4731. // Vodafone specific header: http://www.seoprinciple.com/mobile-web-community-still-angry-at-vodafone/24/
  4732. 'HTTP_X_DEVICE_USER_AGENT',
  4733. 'HTTP_X_ORIGINAL_USER_AGENT',
  4734. 'HTTP_X_SKYFIRE_PHONE',
  4735. 'HTTP_X_BOLT_PHONE_UA',
  4736. 'HTTP_DEVICE_STOCK_UA',
  4737. 'HTTP_X_UCBROWSER_DEVICE_UA'
  4738. );
  4739. /**
  4740. * The individual segments that could exist in a User-Agent string. VER refers to the regular
  4741. * expression defined in the constant self::VER.
  4742. *
  4743. * @var array
  4744. */
  4745. protected static $properties = array(
  4746. // Build
  4747. 'Mobile' => 'Mobile/[VER]',
  4748. 'Build' => 'Build/[VER]',
  4749. 'Version' => 'Version/[VER]',
  4750. 'VendorID' => 'VendorID/[VER]',
  4751. // Devices
  4752. 'iPad' => 'iPad.*CPU[a-z ]+[VER]',
  4753. 'iPhone' => 'iPhone.*CPU[a-z ]+[VER]',
  4754. 'iPod' => 'iPod.*CPU[a-z ]+[VER]',
  4755. //'BlackBerry' => array('BlackBerry[VER]', 'BlackBerry [VER];'),
  4756. 'Kindle' => 'Kindle/[VER]',
  4757. // Browser
  4758. 'Chrome' => array('Chrome/[VER]', 'CriOS/[VER]', 'CrMo/[VER]'),
  4759. 'Coast' => array('Coast/[VER]'),
  4760. 'Dolfin' => 'Dolfin/[VER]',
  4761. // @reference: https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/User-Agent/Firefox
  4762. 'Firefox' => array('Firefox/[VER]', 'FxiOS/[VER]'),
  4763. 'Fennec' => 'Fennec/[VER]',
  4764. // http://msdn.microsoft.com/en-us/library/ms537503(v=vs.85).aspx
  4765. // https://msdn.microsoft.com/en-us/library/ie/hh869301(v=vs.85).aspx
  4766. 'Edge' => 'Edge/[VER]',
  4767. 'IE' => array('IEMobile/[VER];', 'IEMobile [VER]', 'MSIE [VER];', 'Trident/[0-9.]+;.*rv:[VER]'),
  4768. // http://en.wikipedia.org/wiki/NetFront
  4769. 'NetFront' => 'NetFront/[VER]',
  4770. 'NokiaBrowser' => 'NokiaBrowser/[VER]',
  4771. 'Opera' => array( ' OPR/[VER]', 'Opera Mini/[VER]', 'Version/[VER]' ),
  4772. 'Opera Mini' => 'Opera Mini/[VER]',
  4773. 'Opera Mobi' => 'Version/[VER]',
  4774. 'UC Browser' => 'UC Browser[VER]',
  4775. 'MQQBrowser' => 'MQQBrowser/[VER]',
  4776. 'MicroMessenger' => 'MicroMessenger/[VER]',
  4777. 'baiduboxapp' => 'baiduboxapp/[VER]',
  4778. 'baidubrowser' => 'baidubrowser/[VER]',
  4779. 'SamsungBrowser' => 'SamsungBrowser/[VER]',
  4780. 'Iron' => 'Iron/[VER]',
  4781. // @note: Safari 7534.48.3 is actually Version 5.1.
  4782. // @note: On BlackBerry the Version is overwriten by the OS.
  4783. 'Safari' => array( 'Version/[VER]', 'Safari/[VER]' ),
  4784. 'Skyfire' => 'Skyfire/[VER]',
  4785. 'Tizen' => 'Tizen/[VER]',
  4786. 'Webkit' => 'webkit[ /][VER]',
  4787. 'PaleMoon' => 'PaleMoon/[VER]',
  4788. // Engine
  4789. 'Gecko' => 'Gecko/[VER]',
  4790. 'Trident' => 'Trident/[VER]',
  4791. 'Presto' => 'Presto/[VER]',
  4792. 'Goanna' => 'Goanna/[VER]',
  4793. // OS
  4794. 'iOS' => ' \bi?OS\b [VER][ ;]{1}',
  4795. 'Android' => 'Android [VER]',
  4796. 'BlackBerry' => array('BlackBerry[\w]+/[VER]', 'BlackBerry.*Version/[VER]', 'Version/[VER]'),
  4797. 'BREW' => 'BREW [VER]',
  4798. 'Java' => 'Java/[VER]',
  4799. // @reference: http://windowsteamblog.com/windows_phone/b/wpdev/archive/2011/08/29/introducing-the-ie9-on-windows-phone-mango-user-agent-string.aspx
  4800. // @reference: http://en.wikipedia.org/wiki/Windows_NT#Releases
  4801. 'Windows Phone OS' => array( 'Windows Phone OS [VER]', 'Windows Phone [VER]'),
  4802. 'Windows Phone' => 'Windows Phone [VER]',
  4803. 'Windows CE' => 'Windows CE/[VER]',
  4804. // http://social.msdn.microsoft.com/Forums/en-US/windowsdeveloperpreviewgeneral/thread/6be392da-4d2f-41b4-8354-8dcee20c85cd
  4805. 'Windows NT' => 'Windows NT [VER]',
  4806. 'Symbian' => array('SymbianOS/[VER]', 'Symbian/[VER]'),
  4807. 'webOS' => array('webOS/[VER]', 'hpwOS/[VER];'),
  4808. );
  4809. /**
  4810. * Construct an instance of this class.
  4811. *
  4812. * @param array $headers Specify the headers as injection. Should be PHP _SERVER flavored.
  4813. * If left empty, will use the global _SERVER['HTTP_*'] vars instead.
  4814. * @param string $userAgent Inject the User-Agent header. If null, will use HTTP_USER_AGENT
  4815. * from the $headers array instead.
  4816. */
  4817. public function __construct(
  4818. array $headers = null,
  4819. $userAgent = null
  4820. ) {
  4821. $this->setHttpHeaders($headers);
  4822. $this->setUserAgent($userAgent);
  4823. }
  4824. /**
  4825. * Get the current script version.
  4826. * This is useful for the demo.php file,
  4827. * so people can check on what version they are testing
  4828. * for mobile devices.
  4829. *
  4830. * @return string The version number in semantic version format.
  4831. */
  4832. public static function getScriptVersion()
  4833. {
  4834. return self::VERSION;
  4835. }
  4836. /**
  4837. * Set the HTTP Headers. Must be PHP-flavored. This method will reset existing headers.
  4838. *
  4839. * @param array $httpHeaders The headers to set. If null, then using PHP's _SERVER to extract
  4840. * the headers. The default null is left for backwards compatibility.
  4841. */
  4842. public function setHttpHeaders($httpHeaders = null)
  4843. {
  4844. // use global _SERVER if $httpHeaders aren't defined
  4845. if (!is_array($httpHeaders) || !count($httpHeaders)) {
  4846. $httpHeaders = $_SERVER;
  4847. }
  4848. // clear existing headers
  4849. $this->httpHeaders = array();
  4850. // Only save HTTP headers. In PHP land, that means only _SERVER vars that
  4851. // start with HTTP_.
  4852. foreach ($httpHeaders as $key => $value) {
  4853. if (substr($key, 0, 5) === 'HTTP_') {
  4854. $this->httpHeaders[$key] = $value;
  4855. }
  4856. }
  4857. // In case we're dealing with CloudFront, we need to know.
  4858. $this->setCfHeaders($httpHeaders);
  4859. }
  4860. /**
  4861. * Retrieves the HTTP headers.
  4862. *
  4863. * @return array
  4864. */
  4865. public function getHttpHeaders()
  4866. {
  4867. return $this->httpHeaders;
  4868. }
  4869. /**
  4870. * Retrieves a particular header. If it doesn't exist, no exception/error is caused.
  4871. * Simply null is returned.
  4872. *
  4873. * @param string $header The name of the header to retrieve. Can be HTTP compliant such as
  4874. * "User-Agent" or "X-Device-User-Agent" or can be php-esque with the
  4875. * all-caps, HTTP_ prefixed, underscore seperated awesomeness.
  4876. *
  4877. * @return string|null The value of the header.
  4878. */
  4879. public function getHttpHeader($header)
  4880. {
  4881. // are we using PHP-flavored headers?
  4882. if (strpos($header, '_') === false) {
  4883. $header = str_replace('-', '_', $header);
  4884. $header = strtoupper($header);
  4885. }
  4886. // test the alternate, too
  4887. $altHeader = 'HTTP_' . $header;
  4888. //Test both the regular and the HTTP_ prefix
  4889. if (isset($this->httpHeaders[$header])) {
  4890. return $this->httpHeaders[$header];
  4891. } elseif (isset($this->httpHeaders[$altHeader])) {
  4892. return $this->httpHeaders[$altHeader];
  4893. }
  4894. return null;
  4895. }
  4896. public function getMobileHeaders()
  4897. {
  4898. return self::$mobileHeaders;
  4899. }
  4900. /**
  4901. * Get all possible HTTP headers that
  4902. * can contain the User-Agent string.
  4903. *
  4904. * @return array List of HTTP headers.
  4905. */
  4906. public function getUaHttpHeaders()
  4907. {
  4908. return self::$uaHttpHeaders;
  4909. }
  4910. /**
  4911. * Set CloudFront headers
  4912. * http://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/header-caching.html#header-caching-web-device
  4913. *
  4914. * @param array $cfHeaders List of HTTP headers
  4915. *
  4916. * @return boolean If there were CloudFront headers to be set
  4917. */
  4918. public function setCfHeaders($cfHeaders = null) {
  4919. // use global _SERVER if $cfHeaders aren't defined
  4920. if (!is_array($cfHeaders) || !count($cfHeaders)) {
  4921. $cfHeaders = $_SERVER;
  4922. }
  4923. // clear existing headers
  4924. $this->cloudfrontHeaders = array();
  4925. // Only save CLOUDFRONT headers. In PHP land, that means only _SERVER vars that
  4926. // start with cloudfront-.
  4927. $response = false;
  4928. foreach ($cfHeaders as $key => $value) {
  4929. if (substr(strtolower($key), 0, 16) === 'http_cloudfront_') {
  4930. $this->cloudfrontHeaders[strtoupper($key)] = $value;
  4931. $response = true;
  4932. }
  4933. }
  4934. return $response;
  4935. }
  4936. /**
  4937. * Retrieves the cloudfront headers.
  4938. *
  4939. * @return array
  4940. */
  4941. public function getCfHeaders()
  4942. {
  4943. return $this->cloudfrontHeaders;
  4944. }
  4945. /**
  4946. * Set the User-Agent to be used.
  4947. *
  4948. * @param string $userAgent The user agent string to set.
  4949. *
  4950. * @return string|null
  4951. */
  4952. public function setUserAgent($userAgent = null)
  4953. {
  4954. // Invalidate cache due to #375
  4955. $this->cache = array();
  4956. if (false === empty($userAgent)) {
  4957. return $this->userAgent = $userAgent;
  4958. } else {
  4959. $this->userAgent = null;
  4960. foreach ($this->getUaHttpHeaders() as $altHeader) {
  4961. if (false === empty($this->httpHeaders[$altHeader])) { // @todo: should use getHttpHeader(), but it would be slow. (Serban)
  4962. $this->userAgent .= $this->httpHeaders[$altHeader] . " ";
  4963. }
  4964. }
  4965. if (!empty($this->userAgent)) {
  4966. return $this->userAgent = trim($this->userAgent);
  4967. }
  4968. }
  4969. if (count($this->getCfHeaders()) > 0) {
  4970. return $this->userAgent = 'Amazon CloudFront';
  4971. }
  4972. return $this->userAgent = null;
  4973. }
  4974. /**
  4975. * Retrieve the User-Agent.
  4976. *
  4977. * @return string|null The user agent if it's set.
  4978. */
  4979. public function getUserAgent()
  4980. {
  4981. return $this->userAgent;
  4982. }
  4983. /**
  4984. * Set the detection type. Must be one of self::DETECTION_TYPE_MOBILE or
  4985. * self::DETECTION_TYPE_EXTENDED. Otherwise, nothing is set.
  4986. *
  4987. * @deprecated since version 2.6.9
  4988. *
  4989. * @param string $type The type. Must be a self::DETECTION_TYPE_* constant. The default
  4990. * parameter is null which will default to self::DETECTION_TYPE_MOBILE.
  4991. */
  4992. public function setDetectionType($type = null)
  4993. {
  4994. if ($type === null) {
  4995. $type = self::DETECTION_TYPE_MOBILE;
  4996. }
  4997. if ($type !== self::DETECTION_TYPE_MOBILE && $type !== self::DETECTION_TYPE_EXTENDED) {
  4998. return;
  4999. }
  5000. $this->detectionType = $type;
  5001. }
  5002. public function getMatchingRegex()
  5003. {
  5004. return $this->matchingRegex;
  5005. }
  5006. public function getMatchesArray()
  5007. {
  5008. return $this->matchesArray;
  5009. }
  5010. /**
  5011. * Retrieve the list of known phone devices.
  5012. *
  5013. * @return array List of phone devices.
  5014. */
  5015. public static function getPhoneDevices()
  5016. {
  5017. return self::$phoneDevices;
  5018. }
  5019. /**
  5020. * Retrieve the list of known tablet devices.
  5021. *
  5022. * @return array List of tablet devices.
  5023. */
  5024. public static function getTabletDevices()
  5025. {
  5026. return self::$tabletDevices;
  5027. }
  5028. /**
  5029. * Alias for getBrowsers() method.
  5030. *
  5031. * @return array List of user agents.
  5032. */
  5033. public static function getUserAgents()
  5034. {
  5035. return self::getBrowsers();
  5036. }
  5037. /**
  5038. * Retrieve the list of known browsers. Specifically, the user agents.
  5039. *
  5040. * @return array List of browsers / user agents.
  5041. */
  5042. public static function getBrowsers()
  5043. {
  5044. return self::$browsers;
  5045. }
  5046. /**
  5047. * Retrieve the list of known utilities.
  5048. *
  5049. * @return array List of utilities.
  5050. */
  5051. public static function getUtilities()
  5052. {
  5053. return self::$utilities;
  5054. }
  5055. /**
  5056. * Method gets the mobile detection rules. This method is used for the magic methods $detect->is*().
  5057. *
  5058. * @deprecated since version 2.6.9
  5059. *
  5060. * @return array All the rules (but not extended).
  5061. */
  5062. public static function getMobileDetectionRules()
  5063. {
  5064. static $rules;
  5065. if (!$rules) {
  5066. $rules = array_merge(
  5067. self::$phoneDevices,
  5068. self::$tabletDevices,
  5069. self::$operatingSystems,
  5070. self::$browsers
  5071. );
  5072. }
  5073. return $rules;
  5074. }
  5075. /**
  5076. * Method gets the mobile detection rules + utilities.
  5077. * The reason this is separate is because utilities rules
  5078. * don't necessary imply mobile. This method is used inside
  5079. * the new $detect->is('stuff') method.
  5080. *
  5081. * @deprecated since version 2.6.9
  5082. *
  5083. * @return array All the rules + extended.
  5084. */
  5085. public function getMobileDetectionRulesExtended()
  5086. {
  5087. static $rules;
  5088. if (!$rules) {
  5089. // Merge all rules together.
  5090. $rules = array_merge(
  5091. self::$phoneDevices,
  5092. self::$tabletDevices,
  5093. self::$operatingSystems,
  5094. self::$browsers,
  5095. self::$utilities
  5096. );
  5097. }
  5098. return $rules;
  5099. }
  5100. /**
  5101. * Retrieve the current set of rules.
  5102. *
  5103. * @deprecated since version 2.6.9
  5104. *
  5105. * @return array
  5106. */
  5107. public function getRules()
  5108. {
  5109. if ($this->detectionType == self::DETECTION_TYPE_EXTENDED) {
  5110. return self::getMobileDetectionRulesExtended();
  5111. } else {
  5112. return self::getMobileDetectionRules();
  5113. }
  5114. }
  5115. /**
  5116. * Retrieve the list of mobile operating systems.
  5117. *
  5118. * @return array The list of mobile operating systems.
  5119. */
  5120. public static function getOperatingSystems()
  5121. {
  5122. return self::$operatingSystems;
  5123. }
  5124. /**
  5125. * Check the HTTP headers for signs of mobile.
  5126. * This is the fastest mobile check possible; it's used
  5127. * inside isMobile() method.
  5128. *
  5129. * @return bool
  5130. */
  5131. public function checkHttpHeadersForMobile()
  5132. {
  5133. foreach ($this->getMobileHeaders() as $mobileHeader => $matchType) {
  5134. if (isset($this->httpHeaders[$mobileHeader])) {
  5135. if (is_array($matchType['matches'])) {
  5136. foreach ($matchType['matches'] as $_match) {
  5137. if (strpos($this->httpHeaders[$mobileHeader], $_match) !== false) {
  5138. return true;
  5139. }
  5140. }
  5141. return false;
  5142. } else {
  5143. return true;
  5144. }
  5145. }
  5146. }
  5147. return false;
  5148. }
  5149. /**
  5150. * Magic overloading method.
  5151. *
  5152. * @method boolean is[...]()
  5153. * @param string $name
  5154. * @param array $arguments
  5155. * @return mixed
  5156. * @throws BadMethodCallException when the method doesn't exist and doesn't start with 'is'
  5157. */
  5158. public function __call($name, $arguments)
  5159. {
  5160. // make sure the name starts with 'is', otherwise
  5161. if (substr($name, 0, 2) !== 'is') {
  5162. throw new BadMethodCallException("No such method exists: $name");
  5163. }
  5164. $this->setDetectionType(self::DETECTION_TYPE_MOBILE);
  5165. $key = substr($name, 2);
  5166. return $this->matchUAAgainstKey($key);
  5167. }
  5168. /**
  5169. * Find a detection rule that matches the current User-agent.
  5170. *
  5171. * @param null $userAgent deprecated
  5172. * @return boolean
  5173. */
  5174. protected function matchDetectionRulesAgainstUA($userAgent = null)
  5175. {
  5176. // Begin general search.
  5177. foreach ($this->getRules() as $_regex) {
  5178. if (empty($_regex)) {
  5179. continue;
  5180. }
  5181. if ($this->match($_regex, $userAgent)) {
  5182. return true;
  5183. }
  5184. }
  5185. return false;
  5186. }
  5187. /**
  5188. * Search for a certain key in the rules array.
  5189. * If the key is found then try to match the corresponding
  5190. * regex against the User-Agent.
  5191. *
  5192. * @param string $key
  5193. *
  5194. * @return boolean
  5195. */
  5196. protected function matchUAAgainstKey($key)
  5197. {
  5198. // Make the keys lowercase so we can match: isIphone(), isiPhone(), isiphone(), etc.
  5199. $key = strtolower($key);
  5200. if (false === isset($this->cache[$key])) {
  5201. // change the keys to lower case
  5202. $_rules = array_change_key_case($this->getRules());
  5203. if (false === empty($_rules[$key])) {
  5204. $this->cache[$key] = $this->match($_rules[$key]);
  5205. }
  5206. if (false === isset($this->cache[$key])) {
  5207. $this->cache[$key] = false;
  5208. }
  5209. }
  5210. return $this->cache[$key];
  5211. }
  5212. /**
  5213. * Check if the device is mobile.
  5214. * Returns true if any type of mobile device detected, including special ones
  5215. * @param null $userAgent deprecated
  5216. * @param null $httpHeaders deprecated
  5217. * @return bool
  5218. */
  5219. public function isMobile($userAgent = null, $httpHeaders = null)
  5220. {
  5221. if ($httpHeaders) {
  5222. $this->setHttpHeaders($httpHeaders);
  5223. }
  5224. if ($userAgent) {
  5225. $this->setUserAgent($userAgent);
  5226. }
  5227. // Check specifically for cloudfront headers if the useragent === 'Amazon CloudFront'
  5228. if ($this->getUserAgent() === 'Amazon CloudFront') {
  5229. $cfHeaders = $this->getCfHeaders();
  5230. if(array_key_exists('HTTP_CLOUDFRONT_IS_MOBILE_VIEWER', $cfHeaders) && $cfHeaders['HTTP_CLOUDFRONT_IS_MOBILE_VIEWER'] === 'true') {
  5231. return true;
  5232. }
  5233. }
  5234. $this->setDetectionType(self::DETECTION_TYPE_MOBILE);
  5235. if ($this->checkHttpHeadersForMobile()) {
  5236. return true;
  5237. } else {
  5238. return $this->matchDetectionRulesAgainstUA();
  5239. }
  5240. }
  5241. /**
  5242. * Check if the device is a tablet.
  5243. * Return true if any type of tablet device is detected.
  5244. *
  5245. * @param string $userAgent deprecated
  5246. * @param array $httpHeaders deprecated
  5247. * @return bool
  5248. */
  5249. public function isTablet($userAgent = null, $httpHeaders = null)
  5250. {
  5251. // Check specifically for cloudfront headers if the useragent === 'Amazon CloudFront'
  5252. if ($this->getUserAgent() === 'Amazon CloudFront') {
  5253. $cfHeaders = $this->getCfHeaders();
  5254. if(array_key_exists('HTTP_CLOUDFRONT_IS_TABLET_VIEWER', $cfHeaders) && $cfHeaders['HTTP_CLOUDFRONT_IS_TABLET_VIEWER'] === 'true') {
  5255. return true;
  5256. }
  5257. }
  5258. $this->setDetectionType(self::DETECTION_TYPE_MOBILE);
  5259. foreach (self::$tabletDevices as $_regex) {
  5260. if ($this->match($_regex, $userAgent)) {
  5261. return true;
  5262. }
  5263. }
  5264. return false;
  5265. }
  5266. /**
  5267. * This method checks for a certain property in the
  5268. * userAgent.
  5269. * @todo: The httpHeaders part is not yet used.
  5270. *
  5271. * @param string $key
  5272. * @param string $userAgent deprecated
  5273. * @param string $httpHeaders deprecated
  5274. * @return bool|int|null
  5275. */
  5276. public function is($key, $userAgent = null, $httpHeaders = null)
  5277. {
  5278. // Set the UA and HTTP headers only if needed (eg. batch mode).
  5279. if ($httpHeaders) {
  5280. $this->setHttpHeaders($httpHeaders);
  5281. }
  5282. if ($userAgent) {
  5283. $this->setUserAgent($userAgent);
  5284. }
  5285. $this->setDetectionType(self::DETECTION_TYPE_EXTENDED);
  5286. return $this->matchUAAgainstKey($key);
  5287. }
  5288. /**
  5289. * Some detection rules are relative (not standard),
  5290. * because of the diversity of devices, vendors and
  5291. * their conventions in representing the User-Agent or
  5292. * the HTTP headers.
  5293. *
  5294. * This method will be used to check custom regexes against
  5295. * the User-Agent string.
  5296. *
  5297. * @param $regex
  5298. * @param string $userAgent
  5299. * @return bool
  5300. *
  5301. * @todo: search in the HTTP headers too.
  5302. */
  5303. public function match($regex, $userAgent = null)
  5304. {
  5305. $match = (bool) preg_match(sprintf('#%s#is', $regex), (false === empty($userAgent) ? $userAgent : $this->userAgent), $matches);
  5306. // If positive match is found, store the results for debug.
  5307. if ($match) {
  5308. $this->matchingRegex = $regex;
  5309. $this->matchesArray = $matches;
  5310. }
  5311. return $match;
  5312. }
  5313. /**
  5314. * Get the properties array.
  5315. *
  5316. * @return array
  5317. */
  5318. public static function getProperties()
  5319. {
  5320. return self::$properties;
  5321. }
  5322. /**
  5323. * Prepare the version number.
  5324. *
  5325. * @todo Remove the error supression from str_replace() call.
  5326. *
  5327. * @param string $ver The string version, like "2.6.21.2152";
  5328. *
  5329. * @return float
  5330. */
  5331. public function prepareVersionNo($ver)
  5332. {
  5333. $ver = str_replace(array('_', ' ', '/'), '.', $ver);
  5334. $arrVer = explode('.', $ver, 2);
  5335. if (isset($arrVer[1])) {
  5336. $arrVer[1] = @str_replace('.', '', $arrVer[1]); // @todo: treat strings versions.
  5337. }
  5338. return (float) implode('.', $arrVer);
  5339. }
  5340. /**
  5341. * Check the version of the given property in the User-Agent.
  5342. * Will return a float number. (eg. 2_0 will return 2.0, 4.3.1 will return 4.31)
  5343. *
  5344. * @param string $propertyName The name of the property. See self::getProperties() array
  5345. * keys for all possible properties.
  5346. * @param string $type Either self::VERSION_TYPE_STRING to get a string value or
  5347. * self::VERSION_TYPE_FLOAT indicating a float value. This parameter
  5348. * is optional and defaults to self::VERSION_TYPE_STRING. Passing an
  5349. * invalid parameter will default to the this type as well.
  5350. *
  5351. * @return string|float The version of the property we are trying to extract.
  5352. */
  5353. public function version($propertyName, $type = self::VERSION_TYPE_STRING)
  5354. {
  5355. if (empty($propertyName)) {
  5356. return false;
  5357. }
  5358. // set the $type to the default if we don't recognize the type
  5359. if ($type !== self::VERSION_TYPE_STRING && $type !== self::VERSION_TYPE_FLOAT) {
  5360. $type = self::VERSION_TYPE_STRING;
  5361. }
  5362. $properties = self::getProperties();
  5363. // Check if the property exists in the properties array.
  5364. if (true === isset($properties[$propertyName])) {
  5365. // Prepare the pattern to be matched.
  5366. // Make sure we always deal with an array (string is converted).
  5367. $properties[$propertyName] = (array) $properties[$propertyName];
  5368. foreach ($properties[$propertyName] as $propertyMatchString) {
  5369. $propertyPattern = str_replace('[VER]', self::VER, $propertyMatchString);
  5370. // Identify and extract the version.
  5371. preg_match(sprintf('#%s#is', $propertyPattern), $this->userAgent, $match);
  5372. if (false === empty($match[1])) {
  5373. $version = ($type == self::VERSION_TYPE_FLOAT ? $this->prepareVersionNo($match[1]) : $match[1]);
  5374. return $version;
  5375. }
  5376. }
  5377. }
  5378. return false;
  5379. }
  5380. /**
  5381. * Retrieve the mobile grading, using self::MOBILE_GRADE_* constants.
  5382. *
  5383. * @return string One of the self::MOBILE_GRADE_* constants.
  5384. */
  5385. public function mobileGrade()
  5386. {
  5387. $isMobile = $this->isMobile();
  5388. if (
  5389. // Apple iOS 4-7.0 – Tested on the original iPad (4.3 / 5.0), iPad 2 (4.3 / 5.1 / 6.1), iPad 3 (5.1 / 6.0), iPad Mini (6.1), iPad Retina (7.0), iPhone 3GS (4.3), iPhone 4 (4.3 / 5.1), iPhone 4S (5.1 / 6.0), iPhone 5 (6.0), and iPhone 5S (7.0)
  5390. $this->is('iOS') && $this->version('iPad', self::VERSION_TYPE_FLOAT) >= 4.3 ||
  5391. $this->is('iOS') && $this->version('iPhone', self::VERSION_TYPE_FLOAT) >= 4.3 ||
  5392. $this->is('iOS') && $this->version('iPod', self::VERSION_TYPE_FLOAT) >= 4.3 ||
  5393. // Android 2.1-2.3 - Tested on the HTC Incredible (2.2), original Droid (2.2), HTC Aria (2.1), Google Nexus S (2.3). Functional on 1.5 & 1.6 but performance may be sluggish, tested on Google G1 (1.5)
  5394. // Android 3.1 (Honeycomb) - Tested on the Samsung Galaxy Tab 10.1 and Motorola XOOM
  5395. // Android 4.0 (ICS) - Tested on a Galaxy Nexus. Note: transition performance can be poor on upgraded devices
  5396. // Android 4.1 (Jelly Bean) - Tested on a Galaxy Nexus and Galaxy 7
  5397. ( $this->version('Android', self::VERSION_TYPE_FLOAT)>2.1 && $this->is('Webkit') ) ||
  5398. // Windows Phone 7.5-8 - Tested on the HTC Surround (7.5), HTC Trophy (7.5), LG-E900 (7.5), Nokia 800 (7.8), HTC Mazaa (7.8), Nokia Lumia 520 (8), Nokia Lumia 920 (8), HTC 8x (8)
  5399. $this->version('Windows Phone OS', self::VERSION_TYPE_FLOAT) >= 7.5 ||
  5400. // Tested on the Torch 9800 (6) and Style 9670 (6), BlackBerry® Torch 9810 (7), BlackBerry Z10 (10)
  5401. $this->is('BlackBerry') && $this->version('BlackBerry', self::VERSION_TYPE_FLOAT) >= 6.0 ||
  5402. // Blackberry Playbook (1.0-2.0) - Tested on PlayBook
  5403. $this->match('Playbook.*Tablet') ||
  5404. // Palm WebOS (1.4-3.0) - Tested on the Palm Pixi (1.4), Pre (1.4), Pre 2 (2.0), HP TouchPad (3.0)
  5405. ( $this->version('webOS', self::VERSION_TYPE_FLOAT) >= 1.4 && $this->match('Palm|Pre|Pixi') ) ||
  5406. // Palm WebOS 3.0 - Tested on HP TouchPad
  5407. $this->match('hp.*TouchPad') ||
  5408. // Firefox Mobile 18 - Tested on Android 2.3 and 4.1 devices
  5409. ( $this->is('Firefox') && $this->version('Firefox', self::VERSION_TYPE_FLOAT) >= 18 ) ||
  5410. // Chrome for Android - Tested on Android 4.0, 4.1 device
  5411. ( $this->is('Chrome') && $this->is('AndroidOS') && $this->version('Android', self::VERSION_TYPE_FLOAT) >= 4.0 ) ||
  5412. // Skyfire 4.1 - Tested on Android 2.3 device
  5413. ( $this->is('Skyfire') && $this->version('Skyfire', self::VERSION_TYPE_FLOAT) >= 4.1 && $this->is('AndroidOS') && $this->version('Android', self::VERSION_TYPE_FLOAT) >= 2.3 ) ||
  5414. // Opera Mobile 11.5-12: Tested on Android 2.3
  5415. ( $this->is('Opera') && $this->version('Opera Mobi', self::VERSION_TYPE_FLOAT) >= 11.5 && $this->is('AndroidOS') ) ||
  5416. // Meego 1.2 - Tested on Nokia 950 and N9
  5417. $this->is('MeeGoOS') ||
  5418. // Tizen (pre-release) - Tested on early hardware
  5419. $this->is('Tizen') ||
  5420. // Samsung Bada 2.0 - Tested on a Samsung Wave 3, Dolphin browser
  5421. // @todo: more tests here!
  5422. $this->is('Dolfin') && $this->version('Bada', self::VERSION_TYPE_FLOAT) >= 2.0 ||
  5423. // UC Browser - Tested on Android 2.3 device
  5424. ( ($this->is('UC Browser') || $this->is('Dolfin')) && $this->version('Android', self::VERSION_TYPE_FLOAT) >= 2.3 ) ||
  5425. // Kindle 3 and Fire - Tested on the built-in WebKit browser for each
  5426. ( $this->match('Kindle Fire') ||
  5427. $this->is('Kindle') && $this->version('Kindle', self::VERSION_TYPE_FLOAT) >= 3.0 ) ||
  5428. // Nook Color 1.4.1 - Tested on original Nook Color, not Nook Tablet
  5429. $this->is('AndroidOS') && $this->is('NookTablet') ||
  5430. // Chrome Desktop 16-24 - Tested on OS X 10.7 and Windows 7
  5431. $this->version('Chrome', self::VERSION_TYPE_FLOAT) >= 16 && !$isMobile ||
  5432. // Safari Desktop 5-6 - Tested on OS X 10.7 and Windows 7
  5433. $this->version('Safari', self::VERSION_TYPE_FLOAT) >= 5.0 && !$isMobile ||
  5434. // Firefox Desktop 10-18 - Tested on OS X 10.7 and Windows 7
  5435. $this->version('Firefox', self::VERSION_TYPE_FLOAT) >= 10.0 && !$isMobile ||
  5436. // Internet Explorer 7-9 - Tested on Windows XP, Vista and 7
  5437. $this->version('IE', self::VERSION_TYPE_FLOAT) >= 7.0 && !$isMobile ||
  5438. // Opera Desktop 10-12 - Tested on OS X 10.7 and Windows 7
  5439. $this->version('Opera', self::VERSION_TYPE_FLOAT) >= 10 && !$isMobile
  5440. ){
  5441. return self::MOBILE_GRADE_A;
  5442. }
  5443. if (
  5444. $this->is('iOS') && $this->version('iPad', self::VERSION_TYPE_FLOAT)<4.3 ||
  5445. $this->is('iOS') && $this->version('iPhone', self::VERSION_TYPE_FLOAT)<4.3 ||
  5446. $this->is('iOS') && $this->version('iPod', self::VERSION_TYPE_FLOAT)<4.3 ||
  5447. // Blackberry 5.0: Tested on the Storm 2 9550, Bold 9770
  5448. $this->is('Blackberry') && $this->version('BlackBerry', self::VERSION_TYPE_FLOAT) >= 5 && $this->version('BlackBerry', self::VERSION_TYPE_FLOAT)<6 ||
  5449. //Opera Mini (5.0-6.5) - Tested on iOS 3.2/4.3 and Android 2.3
  5450. ($this->version('Opera Mini', self::VERSION_TYPE_FLOAT) >= 5.0 && $this->version('Opera Mini', self::VERSION_TYPE_FLOAT) <= 7.0 &&
  5451. ($this->version('Android', self::VERSION_TYPE_FLOAT) >= 2.3 || $this->is('iOS')) ) ||
  5452. // Nokia Symbian^3 - Tested on Nokia N8 (Symbian^3), C7 (Symbian^3), also works on N97 (Symbian^1)
  5453. $this->match('NokiaN8|NokiaC7|N97.*Series60|Symbian/3') ||
  5454. // @todo: report this (tested on Nokia N71)
  5455. $this->version('Opera Mobi', self::VERSION_TYPE_FLOAT) >= 11 && $this->is('SymbianOS')
  5456. ){
  5457. return self::MOBILE_GRADE_B;
  5458. }
  5459. if (
  5460. // Blackberry 4.x - Tested on the Curve 8330
  5461. $this->version('BlackBerry', self::VERSION_TYPE_FLOAT) <= 5.0 ||
  5462. // Windows Mobile - Tested on the HTC Leo (WinMo 5.2)
  5463. $this->match('MSIEMobile|Windows CE.*Mobile') || $this->version('Windows Mobile', self::VERSION_TYPE_FLOAT) <= 5.2 ||
  5464. // Tested on original iPhone (3.1), iPhone 3 (3.2)
  5465. $this->is('iOS') && $this->version('iPad', self::VERSION_TYPE_FLOAT) <= 3.2 ||
  5466. $this->is('iOS') && $this->version('iPhone', self::VERSION_TYPE_FLOAT) <= 3.2 ||
  5467. $this->is('iOS') && $this->version('iPod', self::VERSION_TYPE_FLOAT) <= 3.2 ||
  5468. // Internet Explorer 7 and older - Tested on Windows XP
  5469. $this->version('IE', self::VERSION_TYPE_FLOAT) <= 7.0 && !$isMobile
  5470. ){
  5471. return self::MOBILE_GRADE_C;
  5472. }
  5473. // All older smartphone platforms and featurephones - Any device that doesn't support media queries
  5474. // will receive the basic, C grade experience.
  5475. return self::MOBILE_GRADE_C;
  5476. }
  5477. }
  5478. $mobileDetect = new Mobile_Detect;
  5479. $userDevice = ($mobileDetect->isMobile() ? ($mobileDetect->isTablet() ? 'tablet' : 'phone') : 'computer');
  5480. function orgEmail($header = "Message From Admin", $title = "Important Message", $user = "Organizr User", $mainMessage = "", $button = null, $buttonURL = null, $subTitle = "", $subMessage = ""){
  5481. $path = getServerPath();
  5482. return '
  5483. <!DOCTYPE HTML PUBLIC "-//W3C//DTD XHTML 1.0 Transitional //EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
  5484. <html xmlns="http://www.w3.org/1999/xhtml" xmlns:v="urn:schemas-microsoft-com:vml" xmlns:o="urn:schemas-microsoft-com:office:office">
  5485. <head>
  5486. <!--[if gte mso 9]><xml>
  5487. <o:OfficeDocumentSettings>
  5488. <o:AllowPNG/>
  5489. <o:PixelsPerInch>96</o:PixelsPerInch>
  5490. </o:OfficeDocumentSettings>
  5491. </xml><![endif]-->
  5492. <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
  5493. <meta name="viewport" content="width=device-width">
  5494. <!--[if !mso]><!-->
  5495. <meta http-equiv="X-UA-Compatible" content="IE=edge">
  5496. <!--<![endif]-->
  5497. <title></title>
  5498. <!--[if !mso]><!-- -->
  5499. <link href="https://fonts.googleapis.com/css?family=Ubuntu" rel="stylesheet" type="text/css">
  5500. <link href="https://fonts.googleapis.com/css?family=Lato" rel="stylesheet" type="text/css">
  5501. <!--<![endif]-->
  5502. <style type="text/css" id="media-query">
  5503. body {
  5504. margin: 0;
  5505. padding: 0;
  5506. }
  5507. table,
  5508. tr,
  5509. td {
  5510. vertical-align: top;
  5511. border-collapse: collapse;
  5512. }
  5513. .ie-browser table,
  5514. .mso-container table {
  5515. table-layout: fixed;
  5516. }
  5517. * {
  5518. line-height: inherit;
  5519. }
  5520. a[x-apple-data-detectors=true] {
  5521. color: inherit !important;
  5522. text-decoration: none !important;
  5523. }
  5524. [owa] .img-container div,
  5525. [owa] .img-container button {
  5526. display: block !important;
  5527. }
  5528. [owa] .fullwidth button {
  5529. width: 100% !important;
  5530. }
  5531. [owa] .block-grid .col {
  5532. display: table-cell;
  5533. float: none !important;
  5534. vertical-align: top;
  5535. }
  5536. .ie-browser .num12,
  5537. .ie-browser .block-grid,
  5538. [owa] .num12,
  5539. [owa] .block-grid {
  5540. width: 615px !important;
  5541. }
  5542. .ExternalClass,
  5543. .ExternalClass p,
  5544. .ExternalClass span,
  5545. .ExternalClass font,
  5546. .ExternalClass td,
  5547. .ExternalClass div {
  5548. line-height: 100%;
  5549. }
  5550. .ie-browser .mixed-two-up .num4,
  5551. [owa] .mixed-two-up .num4 {
  5552. width: 204px !important;
  5553. }
  5554. .ie-browser .mixed-two-up .num8,
  5555. [owa] .mixed-two-up .num8 {
  5556. width: 408px !important;
  5557. }
  5558. .ie-browser .block-grid.two-up .col,
  5559. [owa] .block-grid.two-up .col {
  5560. width: 307px !important;
  5561. }
  5562. .ie-browser .block-grid.three-up .col,
  5563. [owa] .block-grid.three-up .col {
  5564. width: 205px !important;
  5565. }
  5566. .ie-browser .block-grid.four-up .col,
  5567. [owa] .block-grid.four-up .col {
  5568. width: 153px !important;
  5569. }
  5570. .ie-browser .block-grid.five-up .col,
  5571. [owa] .block-grid.five-up .col {
  5572. width: 123px !important;
  5573. }
  5574. .ie-browser .block-grid.six-up .col,
  5575. [owa] .block-grid.six-up .col {
  5576. width: 102px !important;
  5577. }
  5578. .ie-browser .block-grid.seven-up .col,
  5579. [owa] .block-grid.seven-up .col {
  5580. width: 87px !important;
  5581. }
  5582. .ie-browser .block-grid.eight-up .col,
  5583. [owa] .block-grid.eight-up .col {
  5584. width: 76px !important;
  5585. }
  5586. .ie-browser .block-grid.nine-up .col,
  5587. [owa] .block-grid.nine-up .col {
  5588. width: 68px !important;
  5589. }
  5590. .ie-browser .block-grid.ten-up .col,
  5591. [owa] .block-grid.ten-up .col {
  5592. width: 61px !important;
  5593. }
  5594. .ie-browser .block-grid.eleven-up .col,
  5595. [owa] .block-grid.eleven-up .col {
  5596. width: 55px !important;
  5597. }
  5598. .ie-browser .block-grid.twelve-up .col,
  5599. [owa] .block-grid.twelve-up .col {
  5600. width: 51px !important;
  5601. }
  5602. @media only screen and (min-width: 635px) {
  5603. .block-grid {
  5604. width: 615px !important;
  5605. }
  5606. .block-grid .col {
  5607. display: table-cell;
  5608. Float: none !important;
  5609. vertical-align: top;
  5610. }
  5611. .block-grid .col.num12 {
  5612. width: 615px !important;
  5613. }
  5614. .block-grid.mixed-two-up .col.num4 {
  5615. width: 204px !important;
  5616. }
  5617. .block-grid.mixed-two-up .col.num8 {
  5618. width: 408px !important;
  5619. }
  5620. .block-grid.two-up .col {
  5621. width: 307px !important;
  5622. }
  5623. .block-grid.three-up .col {
  5624. width: 205px !important;
  5625. }
  5626. .block-grid.four-up .col {
  5627. width: 153px !important;
  5628. }
  5629. .block-grid.five-up .col {
  5630. width: 123px !important;
  5631. }
  5632. .block-grid.six-up .col {
  5633. width: 102px !important;
  5634. }
  5635. .block-grid.seven-up .col {
  5636. width: 87px !important;
  5637. }
  5638. .block-grid.eight-up .col {
  5639. width: 76px !important;
  5640. }
  5641. .block-grid.nine-up .col {
  5642. width: 68px !important;
  5643. }
  5644. .block-grid.ten-up .col {
  5645. width: 61px !important;
  5646. }
  5647. .block-grid.eleven-up .col {
  5648. width: 55px !important;
  5649. }
  5650. .block-grid.twelve-up .col {
  5651. width: 51px !important;
  5652. }
  5653. }
  5654. @media (max-width: 635px) {
  5655. .block-grid,
  5656. .col {
  5657. min-width: 320px !important;
  5658. max-width: 100% !important;
  5659. }
  5660. .block-grid {
  5661. width: calc(100% - 40px) !important;
  5662. }
  5663. .col {
  5664. width: 100% !important;
  5665. }
  5666. .col>div {
  5667. margin: 0 auto;
  5668. }
  5669. img.fullwidth {
  5670. max-width: 100% !important;
  5671. }
  5672. }
  5673. </style>
  5674. </head>
  5675. <body class="clean-body" style="margin: 0;padding: 0;-webkit-text-size-adjust: 100%;background-color: #FFFFFF">
  5676. <!--[if IE]><div class="ie-browser"><![endif]-->
  5677. <!--[if mso]><div class="mso-container"><![endif]-->
  5678. <div class="nl-container" style="min-width: 320px;Margin: 0 auto;background-color: #FFFFFF">
  5679. <!--[if (mso)|(IE)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td align="center" style="background-color: #FFFFFF;"><![endif]-->
  5680. <div style="background-color:#333333;">
  5681. <div style="Margin: 0 auto;min-width: 320px;max-width: 615px;width: 615px;width: calc(30500% - 193060px);overflow-wrap: break-word;word-wrap: break-word;word-break: break-word;background-color: transparent;"
  5682. class="block-grid ">
  5683. <div style="border-collapse: collapse;display: table;width: 100%;">
  5684. <!--[if (mso)|(IE)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="background-color:#333333;" align="center"><table cellpadding="0" cellspacing="0" border="0" style="width: 615px;"><tr class="layout-full-width" style="background-color:transparent;"><![endif]-->
  5685. <!--[if (mso)|(IE)]><td align="center" width="615" style=" width:615px; padding-right: 0px; padding-left: 0px; padding-top:0px; padding-bottom:0px; border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent;" valign="top"><![endif]-->
  5686. <div class="col num12" style="min-width: 320px;max-width: 615px;width: 615px;width: calc(29500% - 180810px);background-color: transparent;">
  5687. <div style="background-color: transparent; width: 100% !important;">
  5688. <!--[if (!mso)&(!IE)]><!-->
  5689. <div style="border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent; padding-top:0px; padding-bottom:0px; padding-right: 0px; padding-left: 0px;">
  5690. <!--<![endif]-->
  5691. <div align="left" class="img-container left fullwidth" style="padding-right: 30px; padding-left: 30px;">
  5692. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 30px; padding-left: 30px;" align="left"><![endif]-->
  5693. <img class="left fullwidth" align="left" border="0" src="'.$path.'images/organizr-logo-h.png" alt="Image" title="Image"
  5694. style="outline: none;text-decoration: none;-ms-interpolation-mode: bicubic;clear: both;display: block !important;border: 0;height: auto;float: none;width: 100%;max-width: 555px"
  5695. width="555">
  5696. <!--[if mso]></td></tr></table><![endif]-->
  5697. </div>
  5698. <!--[if (!mso)&(!IE)]><!-->
  5699. </div>
  5700. <!--<![endif]-->
  5701. </div>
  5702. </div>
  5703. <!--[if (mso)|(IE)]></td></tr></table></td></tr></table><![endif]-->
  5704. </div>
  5705. </div>
  5706. </div>
  5707. <div style="background-color:#333333;">
  5708. <div style="Margin: 0 auto;min-width: 320px;max-width: 615px;width: 615px;width: calc(30500% - 193060px);overflow-wrap: break-word;word-wrap: break-word;word-break: break-word;background-color: transparent;"
  5709. class="block-grid ">
  5710. <div style="border-collapse: collapse;display: table;width: 100%;">
  5711. <!--[if (mso)|(IE)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="background-color:#333333;" align="center"><table cellpadding="0" cellspacing="0" border="0" style="width: 615px;"><tr class="layout-full-width" style="background-color:transparent;"><![endif]-->
  5712. <!--[if (mso)|(IE)]><td align="center" width="615" style=" width:615px; padding-right: 0px; padding-left: 0px; padding-top:0px; padding-bottom:0px; border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent;" valign="top"><![endif]-->
  5713. <div class="col num12" style="min-width: 320px;max-width: 615px;width: 615px;width: calc(29500% - 180810px);background-color: transparent;">
  5714. <div style="background-color: transparent; width: 100% !important;">
  5715. <!--[if (!mso)&(!IE)]><!-->
  5716. <div style="border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent; padding-top:0px; padding-bottom:0px; padding-right: 0px; padding-left: 0px;">
  5717. <!--<![endif]-->
  5718. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 0px; padding-left: 0px; padding-top: 0px; padding-bottom: 0px;"><![endif]-->
  5719. <div style="font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;line-height:120%;color:#FFFFFF; padding-right: 0px; padding-left: 0px; padding-top: 0px; padding-bottom: 0px;">
  5720. <div style="font-size:12px;line-height:14px;color:#FFFFFF;font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;text-align:left;">
  5721. <p style="margin: 0;font-size: 12px;line-height: 14px;text-align: center"><span style="font-size: 16px; line-height: 19px;"><strong><span style="line-height: 19px; font-size: 16px;">'.$header.'</span></strong>
  5722. </span>
  5723. </p>
  5724. </div>
  5725. </div>
  5726. <!--[if mso]></td></tr></table><![endif]-->
  5727. <!--[if (!mso)&(!IE)]><!-->
  5728. </div>
  5729. <!--<![endif]-->
  5730. </div>
  5731. </div>
  5732. <!--[if (mso)|(IE)]></td></tr></table></td></tr></table><![endif]-->
  5733. </div>
  5734. </div>
  5735. </div>
  5736. <div style="background-color:#393939;">
  5737. <div style="Margin: 0 auto;min-width: 320px;max-width: 615px;width: 615px;width: calc(30500% - 193060px);overflow-wrap: break-word;word-wrap: break-word;word-break: break-word;background-color: transparent;"
  5738. class="block-grid ">
  5739. <div style="border-collapse: collapse;display: table;width: 100%;">
  5740. <!--[if (mso)|(IE)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="background-color:#393939;" align="center"><table cellpadding="0" cellspacing="0" border="0" style="width: 615px;"><tr class="layout-full-width" style="background-color:transparent;"><![endif]-->
  5741. <!--[if (mso)|(IE)]><td align="center" width="615" style=" width:615px; padding-right: 0px; padding-left: 0px; padding-top:5px; padding-bottom:5px; border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent;" valign="top"><![endif]-->
  5742. <div class="col num12" style="min-width: 320px;max-width: 615px;width: 615px;width: calc(29500% - 180810px);background-color: transparent;">
  5743. <div style="background-color: transparent; width: 100% !important;">
  5744. <!--[if (!mso)&(!IE)]><!-->
  5745. <div style="border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent; padding-top:5px; padding-bottom:5px; padding-right: 0px; padding-left: 0px;">
  5746. <!--<![endif]-->
  5747. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 30px; padding-left: 30px; padding-top: 0px; padding-bottom: 0px;"><![endif]-->
  5748. <div style="font-family:\'Ubuntu\', Tahoma, Verdana, Segoe, sans-serif;line-height:120%;color:#FFFFFF; padding-right: 30px; padding-left: 30px; padding-top: 0px; padding-bottom: 0px;">
  5749. <div style="font-family:Ubuntu, Tahoma, Verdana, Segoe, sans-serif;font-size:12px;line-height:14px;color:#FFFFFF;text-align:left;">
  5750. <p style="margin: 0;font-size: 12px;line-height: 14px;text-align: center"><span style="font-size: 16px; line-height: 19px;"><strong>'.$title.'</strong></span></p>
  5751. </div>
  5752. </div>
  5753. <!--[if mso]></td></tr></table><![endif]-->
  5754. <div style="padding-right: 5px; padding-left: 5px; padding-top: 5px; padding-bottom: 5px;">
  5755. <!--[if (mso)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 5px;padding-left: 5px; padding-top: 5px; padding-bottom: 5px;"><table width="55%" align="center" cellpadding="0" cellspacing="0" border="0"><tr><td><![endif]-->
  5756. <div align="center">
  5757. <div style="border-top: 2px solid #66D9EF; width:55%; line-height:2px; height:2px; font-size:2px;">&#160;</div>
  5758. </div>
  5759. <!--[if (mso)]></td></tr></table></td></tr></table><![endif]-->
  5760. </div>
  5761. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 30px; padding-left: 30px; padding-top: 15px; padding-bottom: 10px;"><![endif]-->
  5762. <div style="font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;line-height:120%;color:#FFFFFF; padding-right: 30px; padding-left: 30px; padding-top: 15px; padding-bottom: 10px;">
  5763. <div style="font-family:\'Lato\',Tahoma,Verdana,Segoe,sans-serif;font-size:12px;line-height:14px;color:#FFFFFF;text-align:left;">
  5764. <p style="margin: 0;font-size: 12px;line-height: 14px"><span style="font-size: 28px; line-height: 33px;">Hey '.$user.',</span></p>
  5765. </div>
  5766. </div>
  5767. <!--[if mso]></td></tr></table><![endif]-->
  5768. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 15px; padding-left: 30px; padding-top: 10px; padding-bottom: 25px;"><![endif]-->
  5769. <div style="font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;line-height:180%;color:#FFFFFF; padding-right: 15px; padding-left: 30px; padding-top: 10px; padding-bottom: 25px;">
  5770. <div style="font-size:12px;line-height:22px;font-family:\'Lato\',Tahoma,Verdana,Segoe,sans-serif;color:#FFFFFF;text-align:left;">
  5771. <p style="margin: 0;font-size: 14px;line-height: 25px"><span style="font-size: 18px; line-height: 32px;"><em><span style="line-height: 32px; font-size: 18px;">'.$mainMessage.'</span></em>
  5772. </span>
  5773. </p>
  5774. </div>
  5775. </div>
  5776. <!--[if mso]></td></tr></table><![endif]-->
  5777. <div align="center" class="button-container center" style="padding-right: 30px; padding-left: 30px; padding-top:15px; padding-bottom:15px;">
  5778. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0" style="border-spacing: 0; border-collapse: collapse; mso-table-lspace:0pt; mso-table-rspace:0pt;"><tr><td style="padding-right: 30px; padding-left: 30px; padding-top:15px; padding-bottom:15px;" align="center"><v:roundrect xmlns:v="urn:schemas-microsoft-com:vml" xmlns:w="urn:schemas-microsoft-com:office:word" href="'.$path.'" style="height:48px; v-text-anchor:middle; width:194px;" arcsize="53%" strokecolor="" fillcolor="#66D9EF"><w:anchorlock/><center style="color:#000; font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif; font-size:18px;"><![endif]-->
  5779. <a href="'.$buttonURL.'" target="_blank" style="display: inline-block;text-decoration: none;-webkit-text-size-adjust: none;text-align: center;color: #000; background-color: #66D9EF; border-radius: 25px; -webkit-border-radius: 25px; -moz-border-radius: 25px; max-width: 180px; width: 114px; width: auto; border-top: 3px solid transparent; border-right: 3px solid transparent; border-bottom: 3px solid transparent; border-left: 3px solid transparent; padding-top: 5px; padding-right: 30px; padding-bottom: 5px; padding-left: 30px; font-family: \'Lato\', Tahoma, Verdana, Segoe, sans-serif;mso-border-alt: none">
  5780. <span style="font-size:12px;line-height:21px;"><span style="font-size: 18px; line-height: 32px;" data-mce-style="font-size: 18px; line-height: 44px;">'.$button.'</span></span></a>
  5781. <!--[if mso]></center></v:roundrect></td></tr></table><![endif]-->
  5782. </div>
  5783. <!--[if mso]></center></v:roundrect></td></tr></table><![endif]-->
  5784. </div>
  5785. <!--[if (!mso)&(!IE)]><!-->
  5786. </div>
  5787. <!--<![endif]-->
  5788. </div>
  5789. </div>
  5790. <!--[if (mso)|(IE)]></td></tr></table></td></tr></table><![endif]-->
  5791. </div>
  5792. </div>
  5793. </div>
  5794. <div style="background-color:#ffffff;">
  5795. <div style="Margin: 0 auto;min-width: 320px;max-width: 615px;width: 615px;width: calc(30500% - 193060px);overflow-wrap: break-word;word-wrap: break-word;word-break: break-word;background-color: transparent;"
  5796. class="block-grid ">
  5797. <div style="border-collapse: collapse;display: table;width: 100%;">
  5798. <!--[if (mso)|(IE)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="background-color:#ffffff;" align="center"><table cellpadding="0" cellspacing="0" border="0" style="width: 615px;"><tr class="layout-full-width" style="background-color:transparent;"><![endif]-->
  5799. <!--[if (mso)|(IE)]><td align="center" width="615" style=" width:615px; padding-right: 0px; padding-left: 0px; padding-top:5px; padding-bottom:30px; border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent;" valign="top"><![endif]-->
  5800. <div class="col num12" style="min-width: 320px;max-width: 615px;width: 615px;width: calc(29500% - 180810px);background-color: transparent;">
  5801. <div style="background-color: transparent; width: 100% !important;">
  5802. <!--[if (!mso)&(!IE)]><!-->
  5803. <div style="border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent; padding-top:5px; padding-bottom:30px; padding-right: 0px; padding-left: 0px;">
  5804. <!--<![endif]-->
  5805. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 10px; padding-left: 10px; padding-top: 0px; padding-bottom: 10px;"><![endif]-->
  5806. <div style="font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;line-height:120%;color:#555555; padding-right: 10px; padding-left: 10px; padding-top: 0px; padding-bottom: 10px;">
  5807. <div style="font-size:12px;line-height:14px;color:#555555;font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;text-align:left;">
  5808. <p style="margin: 0;font-size: 14px;line-height: 17px;text-align: center"><strong><span style="font-size: 26px; line-height: 31px;">'.$subTitle.'<br></span></strong></p>
  5809. </div>
  5810. </div>
  5811. <!--[if mso]></td></tr></table><![endif]-->
  5812. <div style="padding-right: 20px; padding-left: 20px; padding-top: 15px; padding-bottom: 20px;">
  5813. <!--[if (mso)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 20px;padding-left: 20px; padding-top: 15px; padding-bottom: 20px;"><table width="40%" align="center" cellpadding="0" cellspacing="0" border="0"><tr><td><![endif]-->
  5814. <div align="center">
  5815. <div style="border-top: 3px solid #66D9EF; width:40%; line-height:3px; height:3px; font-size:3px;">&#160;</div>
  5816. </div>
  5817. <!--[if (mso)]></td></tr></table></td></tr></table><![endif]-->
  5818. </div>
  5819. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 10px; padding-left: 10px; padding-top: 0px; padding-bottom: 0px;"><![endif]-->
  5820. <div style="font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;line-height:180%;color:#7E7D7D; padding-right: 10px; padding-left: 10px; padding-top: 0px; padding-bottom: 0px;">
  5821. <div style="font-size:12px;line-height:22px;color:#7E7D7D;font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;text-align:left;">
  5822. <p style="margin: 0;font-size: 14px;line-height: 25px;text-align: center"><em><span style="font-size: 18px; line-height: 32px;">'.$subMessage.'</span></em></p>
  5823. </div>
  5824. </div>
  5825. <!--[if mso]></td></tr></table><![endif]-->
  5826. <!--[if (!mso)&(!IE)]><!-->
  5827. </div>
  5828. <!--<![endif]-->
  5829. </div>
  5830. </div>
  5831. <!--[if (mso)|(IE)]></td></tr></table></td></tr></table><![endif]-->
  5832. </div>
  5833. </div>
  5834. </div>
  5835. <div style="background-color:#333333;">
  5836. <div style="Margin: 0 auto;min-width: 320px;max-width: 615px;width: 615px;width: calc(30500% - 193060px);overflow-wrap: break-word;word-wrap: break-word;word-break: break-word;background-color: transparent;"
  5837. class="block-grid ">
  5838. <div style="border-collapse: collapse;display: table;width: 100%;">
  5839. <!--[if (mso)|(IE)]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="background-color:#333333;" align="center"><table cellpadding="0" cellspacing="0" border="0" style="width: 615px;"><tr class="layout-full-width" style="background-color:transparent;"><![endif]-->
  5840. <!--[if (mso)|(IE)]><td align="center" width="615" style=" width:615px; padding-right: 0px; padding-left: 0px; padding-top:5px; padding-bottom:5px; border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent;" valign="top"><![endif]-->
  5841. <div class="col num12" style="min-width: 320px;max-width: 615px;width: 615px;width: calc(29500% - 180810px);background-color: transparent;">
  5842. <div style="background-color: transparent; width: 100% !important;">
  5843. <!--[if (!mso)&(!IE)]><!-->
  5844. <div style="border-top: 0px solid transparent; border-left: 0px solid transparent; border-bottom: 0px solid transparent; border-right: 0px solid transparent; padding-top:5px; padding-bottom:5px; padding-right: 0px; padding-left: 0px;">
  5845. <!--<![endif]-->
  5846. <!--[if mso]><table width="100%" cellpadding="0" cellspacing="0" border="0"><tr><td style="padding-right: 10px; padding-left: 10px; padding-top: 10px; padding-bottom: 10px;"><![endif]-->
  5847. <div style="font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;line-height:120%;color:#959595; padding-right: 10px; padding-left: 10px; padding-top: 10px; padding-bottom: 10px;">
  5848. <div style="font-size:12px;line-height:14px;color:#959595;font-family:\'Lato\', Tahoma, Verdana, Segoe, sans-serif;text-align:left;">
  5849. <p style="margin: 0;font-size: 14px;line-height: 17px;text-align: center">This&#160;email was sent by <a style="color:#AD80FD;text-decoration: underline;" title="Organizr"
  5850. href="https://github.com/causefx/Organizr" target="_blank" rel="noopener noreferrer">Organizr</a><strong><br></strong></p>
  5851. </div>
  5852. </div>
  5853. <!--[if mso]></td></tr></table><![endif]-->
  5854. <!--[if (!mso)&(!IE)]><!-->
  5855. </div>
  5856. <!--<![endif]-->
  5857. </div>
  5858. </div>
  5859. <!--[if (mso)|(IE)]></td></tr></table></td></tr></table><![endif]-->
  5860. </div>
  5861. </div>
  5862. </div>
  5863. <!--[if (mso)|(IE)]></td></tr></table><![endif]-->
  5864. </div>
  5865. <!--[if (mso)|(IE)]></div><![endif]-->
  5866. </body>
  5867. </html>
  5868. ';
  5869. }
  5870. function mimeTypes(){
  5871. return array(
  5872. '123' => 'application/vnd.lotus-1-2-3',
  5873. '3dml' => 'text/vnd.in3d.3dml',
  5874. '3ds' => 'image/x-3ds',
  5875. '3g2' => 'video/3gpp2',
  5876. '3gp' => 'video/3gpp',
  5877. '7z' => 'application/x-7z-compressed',
  5878. 'aab' => 'application/x-authorware-bin',
  5879. 'aac' => 'audio/x-aac',
  5880. 'aam' => 'application/x-authorware-map',
  5881. 'aas' => 'application/x-authorware-seg',
  5882. 'abw' => 'application/x-abiword',
  5883. 'ac' => 'application/pkix-attr-cert',
  5884. 'acc' => 'application/vnd.americandynamics.acc',
  5885. 'ace' => 'application/x-ace-compressed',
  5886. 'acu' => 'application/vnd.acucobol',
  5887. 'acutc' => 'application/vnd.acucorp',
  5888. 'adp' => 'audio/adpcm',
  5889. 'aep' => 'application/vnd.audiograph',
  5890. 'afm' => 'application/x-font-type1',
  5891. 'afp' => 'application/vnd.ibm.modcap',
  5892. 'ahead' => 'application/vnd.ahead.space',
  5893. 'ai' => 'application/postscript',
  5894. 'aif' => 'audio/x-aiff',
  5895. 'aifc' => 'audio/x-aiff',
  5896. 'aiff' => 'audio/x-aiff',
  5897. 'air' => 'application/vnd.adobe.air-application-installer-package+zip',
  5898. 'ait' => 'application/vnd.dvb.ait',
  5899. 'ami' => 'application/vnd.amiga.ami',
  5900. 'apk' => 'application/vnd.android.package-archive',
  5901. 'appcache' => 'text/cache-manifest',
  5902. 'application' => 'application/x-ms-application',
  5903. 'apr' => 'application/vnd.lotus-approach',
  5904. 'arc' => 'application/x-freearc',
  5905. 'asc' => 'application/pgp-signature',
  5906. 'asf' => 'video/x-ms-asf',
  5907. 'asm' => 'text/x-asm',
  5908. 'aso' => 'application/vnd.accpac.simply.aso',
  5909. 'asx' => 'video/x-ms-asf',
  5910. 'atc' => 'application/vnd.acucorp',
  5911. 'atom' => 'application/atom+xml',
  5912. 'atomcat' => 'application/atomcat+xml',
  5913. 'atomsvc' => 'application/atomsvc+xml',
  5914. 'atx' => 'application/vnd.antix.game-component',
  5915. 'au' => 'audio/basic',
  5916. 'avi' => 'video/x-msvideo',
  5917. 'aw' => 'application/applixware',
  5918. 'azf' => 'application/vnd.airzip.filesecure.azf',
  5919. 'azs' => 'application/vnd.airzip.filesecure.azs',
  5920. 'azw' => 'application/vnd.amazon.ebook',
  5921. 'bat' => 'application/x-msdownload',
  5922. 'bcpio' => 'application/x-bcpio',
  5923. 'bdf' => 'application/x-font-bdf',
  5924. 'bdm' => 'application/vnd.syncml.dm+wbxml',
  5925. 'bed' => 'application/vnd.realvnc.bed',
  5926. 'bh2' => 'application/vnd.fujitsu.oasysprs',
  5927. 'bin' => 'application/octet-stream',
  5928. 'blb' => 'application/x-blorb',
  5929. 'blorb' => 'application/x-blorb',
  5930. 'bmi' => 'application/vnd.bmi',
  5931. 'bmp' => 'image/bmp',
  5932. 'book' => 'application/vnd.framemaker',
  5933. 'box' => 'application/vnd.previewsystems.box',
  5934. 'boz' => 'application/x-bzip2',
  5935. 'bpk' => 'application/octet-stream',
  5936. 'btif' => 'image/prs.btif',
  5937. 'bz' => 'application/x-bzip',
  5938. 'bz2' => 'application/x-bzip2',
  5939. 'c' => 'text/x-c',
  5940. 'c11amc' => 'application/vnd.cluetrust.cartomobile-config',
  5941. 'c11amz' => 'application/vnd.cluetrust.cartomobile-config-pkg',
  5942. 'c4d' => 'application/vnd.clonk.c4group',
  5943. 'c4f' => 'application/vnd.clonk.c4group',
  5944. 'c4g' => 'application/vnd.clonk.c4group',
  5945. 'c4p' => 'application/vnd.clonk.c4group',
  5946. 'c4u' => 'application/vnd.clonk.c4group',
  5947. 'cab' => 'application/vnd.ms-cab-compressed',
  5948. 'caf' => 'audio/x-caf',
  5949. 'cap' => 'application/vnd.tcpdump.pcap',
  5950. 'car' => 'application/vnd.curl.car',
  5951. 'cat' => 'application/vnd.ms-pki.seccat',
  5952. 'cb7' => 'application/x-cbr',
  5953. 'cba' => 'application/x-cbr',
  5954. 'cbr' => 'application/x-cbr',
  5955. 'cbt' => 'application/x-cbr',
  5956. 'cbz' => 'application/x-cbr',
  5957. 'cc' => 'text/x-c',
  5958. 'cct' => 'application/x-director',
  5959. 'ccxml' => 'application/ccxml+xml',
  5960. 'cdbcmsg' => 'application/vnd.contact.cmsg',
  5961. 'cdf' => 'application/x-netcdf',
  5962. 'cdkey' => 'application/vnd.mediastation.cdkey',
  5963. 'cdmia' => 'application/cdmi-capability',
  5964. 'cdmic' => 'application/cdmi-container',
  5965. 'cdmid' => 'application/cdmi-domain',
  5966. 'cdmio' => 'application/cdmi-object',
  5967. 'cdmiq' => 'application/cdmi-queue',
  5968. 'cdx' => 'chemical/x-cdx',
  5969. 'cdxml' => 'application/vnd.chemdraw+xml',
  5970. 'cdy' => 'application/vnd.cinderella',
  5971. 'cer' => 'application/pkix-cert',
  5972. 'cfs' => 'application/x-cfs-compressed',
  5973. 'cgm' => 'image/cgm',
  5974. 'chat' => 'application/x-chat',
  5975. 'chm' => 'application/vnd.ms-htmlhelp',
  5976. 'chrt' => 'application/vnd.kde.kchart',
  5977. 'cif' => 'chemical/x-cif',
  5978. 'cii' => 'application/vnd.anser-web-certificate-issue-initiation',
  5979. 'cil' => 'application/vnd.ms-artgalry',
  5980. 'cla' => 'application/vnd.claymore',
  5981. 'class' => 'application/java-vm',
  5982. 'clkk' => 'application/vnd.crick.clicker.keyboard',
  5983. 'clkp' => 'application/vnd.crick.clicker.palette',
  5984. 'clkt' => 'application/vnd.crick.clicker.template',
  5985. 'clkw' => 'application/vnd.crick.clicker.wordbank',
  5986. 'clkx' => 'application/vnd.crick.clicker',
  5987. 'clp' => 'application/x-msclip',
  5988. 'cmc' => 'application/vnd.cosmocaller',
  5989. 'cmdf' => 'chemical/x-cmdf',
  5990. 'cml' => 'chemical/x-cml',
  5991. 'cmp' => 'application/vnd.yellowriver-custom-menu',
  5992. 'cmx' => 'image/x-cmx',
  5993. 'cod' => 'application/vnd.rim.cod',
  5994. 'com' => 'application/x-msdownload',
  5995. 'conf' => 'text/plain',
  5996. 'cpio' => 'application/x-cpio',
  5997. 'cpp' => 'text/x-c',
  5998. 'cpt' => 'application/mac-compactpro',
  5999. 'crd' => 'application/x-mscardfile',
  6000. 'crl' => 'application/pkix-crl',
  6001. 'crt' => 'application/x-x509-ca-cert',
  6002. 'cryptonote' => 'application/vnd.rig.cryptonote',
  6003. 'csh' => 'application/x-csh',
  6004. 'csml' => 'chemical/x-csml',
  6005. 'csp' => 'application/vnd.commonspace',
  6006. 'css' => 'text/css',
  6007. 'cst' => 'application/x-director',
  6008. 'csv' => 'text/csv',
  6009. 'cu' => 'application/cu-seeme',
  6010. 'curl' => 'text/vnd.curl',
  6011. 'cww' => 'application/prs.cww',
  6012. 'cxt' => 'application/x-director',
  6013. 'cxx' => 'text/x-c',
  6014. 'dae' => 'model/vnd.collada+xml',
  6015. 'daf' => 'application/vnd.mobius.daf',
  6016. 'dart' => 'application/vnd.dart',
  6017. 'dataless' => 'application/vnd.fdsn.seed',
  6018. 'davmount' => 'application/davmount+xml',
  6019. 'dbk' => 'application/docbook+xml',
  6020. 'dcr' => 'application/x-director',
  6021. 'dcurl' => 'text/vnd.curl.dcurl',
  6022. 'dd2' => 'application/vnd.oma.dd2+xml',
  6023. 'ddd' => 'application/vnd.fujixerox.ddd',
  6024. 'deb' => 'application/x-debian-package',
  6025. 'def' => 'text/plain',
  6026. 'deploy' => 'application/octet-stream',
  6027. 'der' => 'application/x-x509-ca-cert',
  6028. 'dfac' => 'application/vnd.dreamfactory',
  6029. 'dgc' => 'application/x-dgc-compressed',
  6030. 'dic' => 'text/x-c',
  6031. 'dir' => 'application/x-director',
  6032. 'dis' => 'application/vnd.mobius.dis',
  6033. 'dist' => 'application/octet-stream',
  6034. 'distz' => 'application/octet-stream',
  6035. 'djv' => 'image/vnd.djvu',
  6036. 'djvu' => 'image/vnd.djvu',
  6037. 'dll' => 'application/x-msdownload',
  6038. 'dmg' => 'application/x-apple-diskimage',
  6039. 'dmp' => 'application/vnd.tcpdump.pcap',
  6040. 'dms' => 'application/octet-stream',
  6041. 'dna' => 'application/vnd.dna',
  6042. 'doc' => 'application/msword',
  6043. 'docm' => 'application/vnd.ms-word.document.macroenabled.12',
  6044. 'docx' => 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
  6045. 'dot' => 'application/msword',
  6046. 'dotm' => 'application/vnd.ms-word.template.macroenabled.12',
  6047. 'dotx' => 'application/vnd.openxmlformats-officedocument.wordprocessingml.template',
  6048. 'dp' => 'application/vnd.osgi.dp',
  6049. 'dpg' => 'application/vnd.dpgraph',
  6050. 'dra' => 'audio/vnd.dra',
  6051. 'dsc' => 'text/prs.lines.tag',
  6052. 'dssc' => 'application/dssc+der',
  6053. 'dtb' => 'application/x-dtbook+xml',
  6054. 'dtd' => 'application/xml-dtd',
  6055. 'dts' => 'audio/vnd.dts',
  6056. 'dtshd' => 'audio/vnd.dts.hd',
  6057. 'dump' => 'application/octet-stream',
  6058. 'dvb' => 'video/vnd.dvb.file',
  6059. 'dvi' => 'application/x-dvi',
  6060. 'dwf' => 'model/vnd.dwf',
  6061. 'dwg' => 'image/vnd.dwg',
  6062. 'dxf' => 'image/vnd.dxf',
  6063. 'dxp' => 'application/vnd.spotfire.dxp',
  6064. 'dxr' => 'application/x-director',
  6065. 'ecelp4800' => 'audio/vnd.nuera.ecelp4800',
  6066. 'ecelp7470' => 'audio/vnd.nuera.ecelp7470',
  6067. 'ecelp9600' => 'audio/vnd.nuera.ecelp9600',
  6068. 'ecma' => 'application/ecmascript',
  6069. 'edm' => 'application/vnd.novadigm.edm',
  6070. 'edx' => 'application/vnd.novadigm.edx',
  6071. 'efif' => 'application/vnd.picsel',
  6072. 'ei6' => 'application/vnd.pg.osasli',
  6073. 'elc' => 'application/octet-stream',
  6074. 'emf' => 'application/x-msmetafile',
  6075. 'eml' => 'message/rfc822',
  6076. 'emma' => 'application/emma+xml',
  6077. 'emz' => 'application/x-msmetafile',
  6078. 'eol' => 'audio/vnd.digital-winds',
  6079. 'eot' => 'application/vnd.ms-fontobject',
  6080. 'eps' => 'application/postscript',
  6081. 'epub' => 'application/epub+zip',
  6082. 'es3' => 'application/vnd.eszigno3+xml',
  6083. 'esa' => 'application/vnd.osgi.subsystem',
  6084. 'esf' => 'application/vnd.epson.esf',
  6085. 'et3' => 'application/vnd.eszigno3+xml',
  6086. 'etx' => 'text/x-setext',
  6087. 'eva' => 'application/x-eva',
  6088. 'evy' => 'application/x-envoy',
  6089. 'exe' => 'application/x-msdownload',
  6090. 'exi' => 'application/exi',
  6091. 'ext' => 'application/vnd.novadigm.ext',
  6092. 'ez' => 'application/andrew-inset',
  6093. 'ez2' => 'application/vnd.ezpix-album',
  6094. 'ez3' => 'application/vnd.ezpix-package',
  6095. 'f' => 'text/x-fortran',
  6096. 'f4v' => 'video/x-f4v',
  6097. 'f77' => 'text/x-fortran',
  6098. 'f90' => 'text/x-fortran',
  6099. 'fbs' => 'image/vnd.fastbidsheet',
  6100. 'fcdt' => 'application/vnd.adobe.formscentral.fcdt',
  6101. 'fcs' => 'application/vnd.isac.fcs',
  6102. 'fdf' => 'application/vnd.fdf',
  6103. 'fe_launch' => 'application/vnd.denovo.fcselayout-link',
  6104. 'fg5' => 'application/vnd.fujitsu.oasysgp',
  6105. 'fgd' => 'application/x-director',
  6106. 'fh' => 'image/x-freehand',
  6107. 'fh4' => 'image/x-freehand',
  6108. 'fh5' => 'image/x-freehand',
  6109. 'fh7' => 'image/x-freehand',
  6110. 'fhc' => 'image/x-freehand',
  6111. 'fig' => 'application/x-xfig',
  6112. 'flac' => 'audio/x-flac',
  6113. 'fli' => 'video/x-fli',
  6114. 'flo' => 'application/vnd.micrografx.flo',
  6115. 'flv' => 'video/x-flv',
  6116. 'flw' => 'application/vnd.kde.kivio',
  6117. 'flx' => 'text/vnd.fmi.flexstor',
  6118. 'fly' => 'text/vnd.fly',
  6119. 'fm' => 'application/vnd.framemaker',
  6120. 'fnc' => 'application/vnd.frogans.fnc',
  6121. 'for' => 'text/x-fortran',
  6122. 'fpx' => 'image/vnd.fpx',
  6123. 'frame' => 'application/vnd.framemaker',
  6124. 'fsc' => 'application/vnd.fsc.weblaunch',
  6125. 'fst' => 'image/vnd.fst',
  6126. 'ftc' => 'application/vnd.fluxtime.clip',
  6127. 'fti' => 'application/vnd.anser-web-funds-transfer-initiation',
  6128. 'fvt' => 'video/vnd.fvt',
  6129. 'fxp' => 'application/vnd.adobe.fxp',
  6130. 'fxpl' => 'application/vnd.adobe.fxp',
  6131. 'fzs' => 'application/vnd.fuzzysheet',
  6132. 'g2w' => 'application/vnd.geoplan',
  6133. 'g3' => 'image/g3fax',
  6134. 'g3w' => 'application/vnd.geospace',
  6135. 'gac' => 'application/vnd.groove-account',
  6136. 'gam' => 'application/x-tads',
  6137. 'gbr' => 'application/rpki-ghostbusters',
  6138. 'gca' => 'application/x-gca-compressed',
  6139. 'gdl' => 'model/vnd.gdl',
  6140. 'geo' => 'application/vnd.dynageo',
  6141. 'gex' => 'application/vnd.geometry-explorer',
  6142. 'ggb' => 'application/vnd.geogebra.file',
  6143. 'ggt' => 'application/vnd.geogebra.tool',
  6144. 'ghf' => 'application/vnd.groove-help',
  6145. 'gif' => 'image/gif',
  6146. 'gim' => 'application/vnd.groove-identity-message',
  6147. 'gml' => 'application/gml+xml',
  6148. 'gmx' => 'application/vnd.gmx',
  6149. 'gnumeric' => 'application/x-gnumeric',
  6150. 'gph' => 'application/vnd.flographit',
  6151. 'gpx' => 'application/gpx+xml',
  6152. 'gqf' => 'application/vnd.grafeq',
  6153. 'gqs' => 'application/vnd.grafeq',
  6154. 'gram' => 'application/srgs',
  6155. 'gramps' => 'application/x-gramps-xml',
  6156. 'gre' => 'application/vnd.geometry-explorer',
  6157. 'grv' => 'application/vnd.groove-injector',
  6158. 'grxml' => 'application/srgs+xml',
  6159. 'gsf' => 'application/x-font-ghostscript',
  6160. 'gtar' => 'application/x-gtar',
  6161. 'gtm' => 'application/vnd.groove-tool-message',
  6162. 'gtw' => 'model/vnd.gtw',
  6163. 'gv' => 'text/vnd.graphviz',
  6164. 'gxf' => 'application/gxf',
  6165. 'gxt' => 'application/vnd.geonext',
  6166. 'h' => 'text/x-c',
  6167. 'h261' => 'video/h261',
  6168. 'h263' => 'video/h263',
  6169. 'h264' => 'video/h264',
  6170. 'hal' => 'application/vnd.hal+xml',
  6171. 'hbci' => 'application/vnd.hbci',
  6172. 'hdf' => 'application/x-hdf',
  6173. 'hh' => 'text/x-c',
  6174. 'hlp' => 'application/winhlp',
  6175. 'hpgl' => 'application/vnd.hp-hpgl',
  6176. 'hpid' => 'application/vnd.hp-hpid',
  6177. 'hps' => 'application/vnd.hp-hps',
  6178. 'hqx' => 'application/mac-binhex40',
  6179. 'htke' => 'application/vnd.kenameaapp',
  6180. 'htm' => 'text/html',
  6181. 'html' => 'text/html',
  6182. 'hvd' => 'application/vnd.yamaha.hv-dic',
  6183. 'hvp' => 'application/vnd.yamaha.hv-voice',
  6184. 'hvs' => 'application/vnd.yamaha.hv-script',
  6185. 'i2g' => 'application/vnd.intergeo',
  6186. 'icc' => 'application/vnd.iccprofile',
  6187. 'ice' => 'x-conference/x-cooltalk',
  6188. 'icm' => 'application/vnd.iccprofile',
  6189. 'ico' => 'image/x-icon',
  6190. 'ics' => 'text/calendar',
  6191. 'ief' => 'image/ief',
  6192. 'ifb' => 'text/calendar',
  6193. 'ifm' => 'application/vnd.shana.informed.formdata',
  6194. 'iges' => 'model/iges',
  6195. 'igl' => 'application/vnd.igloader',
  6196. 'igm' => 'application/vnd.insors.igm',
  6197. 'igs' => 'model/iges',
  6198. 'igx' => 'application/vnd.micrografx.igx',
  6199. 'iif' => 'application/vnd.shana.informed.interchange',
  6200. 'imp' => 'application/vnd.accpac.simply.imp',
  6201. 'ims' => 'application/vnd.ms-ims',
  6202. 'in' => 'text/plain',
  6203. 'ink' => 'application/inkml+xml',
  6204. 'inkml' => 'application/inkml+xml',
  6205. 'install' => 'application/x-install-instructions',
  6206. 'iota' => 'application/vnd.astraea-software.iota',
  6207. 'ipfix' => 'application/ipfix',
  6208. 'ipk' => 'application/vnd.shana.informed.package',
  6209. 'irm' => 'application/vnd.ibm.rights-management',
  6210. 'irp' => 'application/vnd.irepository.package+xml',
  6211. 'iso' => 'application/x-iso9660-image',
  6212. 'itp' => 'application/vnd.shana.informed.formtemplate',
  6213. 'ivp' => 'application/vnd.immervision-ivp',
  6214. 'ivu' => 'application/vnd.immervision-ivu',
  6215. 'jad' => 'text/vnd.sun.j2me.app-descriptor',
  6216. 'jam' => 'application/vnd.jam',
  6217. 'jar' => 'application/java-archive',
  6218. 'java' => 'text/x-java-source',
  6219. 'jisp' => 'application/vnd.jisp',
  6220. 'jlt' => 'application/vnd.hp-jlyt',
  6221. 'jnlp' => 'application/x-java-jnlp-file',
  6222. 'joda' => 'application/vnd.joost.joda-archive',
  6223. 'jpe' => 'image/jpeg',
  6224. 'jpeg' => 'image/jpeg',
  6225. 'jpg' => 'image/jpeg',
  6226. 'jpgm' => 'video/jpm',
  6227. 'jpgv' => 'video/jpeg',
  6228. 'jpm' => 'video/jpm',
  6229. 'js' => 'application/javascript',
  6230. 'json' => 'application/json',
  6231. 'jsonml' => 'application/jsonml+json',
  6232. 'kar' => 'audio/midi',
  6233. 'karbon' => 'application/vnd.kde.karbon',
  6234. 'kfo' => 'application/vnd.kde.kformula',
  6235. 'kia' => 'application/vnd.kidspiration',
  6236. 'kml' => 'application/vnd.google-earth.kml+xml',
  6237. 'kmz' => 'application/vnd.google-earth.kmz',
  6238. 'kne' => 'application/vnd.kinar',
  6239. 'knp' => 'application/vnd.kinar',
  6240. 'kon' => 'application/vnd.kde.kontour',
  6241. 'kpr' => 'application/vnd.kde.kpresenter',
  6242. 'kpt' => 'application/vnd.kde.kpresenter',
  6243. 'kpxx' => 'application/vnd.ds-keypoint',
  6244. 'ksp' => 'application/vnd.kde.kspread',
  6245. 'ktr' => 'application/vnd.kahootz',
  6246. 'ktx' => 'image/ktx',
  6247. 'ktz' => 'application/vnd.kahootz',
  6248. 'kwd' => 'application/vnd.kde.kword',
  6249. 'kwt' => 'application/vnd.kde.kword',
  6250. 'lasxml' => 'application/vnd.las.las+xml',
  6251. 'latex' => 'application/x-latex',
  6252. 'lbd' => 'application/vnd.llamagraphics.life-balance.desktop',
  6253. 'lbe' => 'application/vnd.llamagraphics.life-balance.exchange+xml',
  6254. 'les' => 'application/vnd.hhe.lesson-player',
  6255. 'lha' => 'application/x-lzh-compressed',
  6256. 'link66' => 'application/vnd.route66.link66+xml',
  6257. 'list' => 'text/plain',
  6258. 'list3820' => 'application/vnd.ibm.modcap',
  6259. 'listafp' => 'application/vnd.ibm.modcap',
  6260. 'lnk' => 'application/x-ms-shortcut',
  6261. 'log' => 'text/plain',
  6262. 'lostxml' => 'application/lost+xml',
  6263. 'lrf' => 'application/octet-stream',
  6264. 'lrm' => 'application/vnd.ms-lrm',
  6265. 'ltf' => 'application/vnd.frogans.ltf',
  6266. 'lvp' => 'audio/vnd.lucent.voice',
  6267. 'lwp' => 'application/vnd.lotus-wordpro',
  6268. 'lzh' => 'application/x-lzh-compressed',
  6269. 'm13' => 'application/x-msmediaview',
  6270. 'm14' => 'application/x-msmediaview',
  6271. 'm1v' => 'video/mpeg',
  6272. 'm21' => 'application/mp21',
  6273. 'm2a' => 'audio/mpeg',
  6274. 'm2v' => 'video/mpeg',
  6275. 'm3a' => 'audio/mpeg',
  6276. 'm3u' => 'audio/x-mpegurl',
  6277. 'm3u8' => 'application/vnd.apple.mpegurl',
  6278. 'm4a' => 'audio/mp4',
  6279. 'm4u' => 'video/vnd.mpegurl',
  6280. 'm4v' => 'video/x-m4v',
  6281. 'ma' => 'application/mathematica',
  6282. 'mads' => 'application/mads+xml',
  6283. 'mag' => 'application/vnd.ecowin.chart',
  6284. 'maker' => 'application/vnd.framemaker',
  6285. 'man' => 'text/troff',
  6286. 'mar' => 'application/octet-stream',
  6287. 'mathml' => 'application/mathml+xml',
  6288. 'mb' => 'application/mathematica',
  6289. 'mbk' => 'application/vnd.mobius.mbk',
  6290. 'mbox' => 'application/mbox',
  6291. 'mc1' => 'application/vnd.medcalcdata',
  6292. 'mcd' => 'application/vnd.mcd',
  6293. 'mcurl' => 'text/vnd.curl.mcurl',
  6294. 'mdb' => 'application/x-msaccess',
  6295. 'mdi' => 'image/vnd.ms-modi',
  6296. 'me' => 'text/troff',
  6297. 'mesh' => 'model/mesh',
  6298. 'meta4' => 'application/metalink4+xml',
  6299. 'metalink' => 'application/metalink+xml',
  6300. 'mets' => 'application/mets+xml',
  6301. 'mfm' => 'application/vnd.mfmp',
  6302. 'mft' => 'application/rpki-manifest',
  6303. 'mgp' => 'application/vnd.osgeo.mapguide.package',
  6304. 'mgz' => 'application/vnd.proteus.magazine',
  6305. 'mid' => 'audio/midi',
  6306. 'midi' => 'audio/midi',
  6307. 'mie' => 'application/x-mie',
  6308. 'mif' => 'application/vnd.mif',
  6309. 'mime' => 'message/rfc822',
  6310. 'mj2' => 'video/mj2',
  6311. 'mjp2' => 'video/mj2',
  6312. 'mk3d' => 'video/x-matroska',
  6313. 'mka' => 'audio/x-matroska',
  6314. 'mks' => 'video/x-matroska',
  6315. 'mkv' => 'video/x-matroska',
  6316. 'mlp' => 'application/vnd.dolby.mlp',
  6317. 'mmd' => 'application/vnd.chipnuts.karaoke-mmd',
  6318. 'mmf' => 'application/vnd.smaf',
  6319. 'mmr' => 'image/vnd.fujixerox.edmics-mmr',
  6320. 'mng' => 'video/x-mng',
  6321. 'mny' => 'application/x-msmoney',
  6322. 'mobi' => 'application/x-mobipocket-ebook',
  6323. 'mods' => 'application/mods+xml',
  6324. 'mov' => 'video/quicktime',
  6325. 'movie' => 'video/x-sgi-movie',
  6326. 'mp2' => 'audio/mpeg',
  6327. 'mp21' => 'application/mp21',
  6328. 'mp2a' => 'audio/mpeg',
  6329. 'mp3' => 'audio/mpeg',
  6330. 'mp4' => 'video/mp4',
  6331. 'mp4a' => 'audio/mp4',
  6332. 'mp4s' => 'application/mp4',
  6333. 'mp4v' => 'video/mp4',
  6334. 'mpc' => 'application/vnd.mophun.certificate',
  6335. 'mpe' => 'video/mpeg',
  6336. 'mpeg' => 'video/mpeg',
  6337. 'mpg' => 'video/mpeg',
  6338. 'mpg4' => 'video/mp4',
  6339. 'mpga' => 'audio/mpeg',
  6340. 'mpkg' => 'application/vnd.apple.installer+xml',
  6341. 'mpm' => 'application/vnd.blueice.multipass',
  6342. 'mpn' => 'application/vnd.mophun.application',
  6343. 'mpp' => 'application/vnd.ms-project',
  6344. 'mpt' => 'application/vnd.ms-project',
  6345. 'mpy' => 'application/vnd.ibm.minipay',
  6346. 'mqy' => 'application/vnd.mobius.mqy',
  6347. 'mrc' => 'application/marc',
  6348. 'mrcx' => 'application/marcxml+xml',
  6349. 'ms' => 'text/troff',
  6350. 'mscml' => 'application/mediaservercontrol+xml',
  6351. 'mseed' => 'application/vnd.fdsn.mseed',
  6352. 'mseq' => 'application/vnd.mseq',
  6353. 'msf' => 'application/vnd.epson.msf',
  6354. 'msh' => 'model/mesh',
  6355. 'msi' => 'application/x-msdownload',
  6356. 'msl' => 'application/vnd.mobius.msl',
  6357. 'msty' => 'application/vnd.muvee.style',
  6358. 'mts' => 'model/vnd.mts',
  6359. 'mus' => 'application/vnd.musician',
  6360. 'musicxml' => 'application/vnd.recordare.musicxml+xml',
  6361. 'mvb' => 'application/x-msmediaview',
  6362. 'mwf' => 'application/vnd.mfer',
  6363. 'mxf' => 'application/mxf',
  6364. 'mxl' => 'application/vnd.recordare.musicxml',
  6365. 'mxml' => 'application/xv+xml',
  6366. 'mxs' => 'application/vnd.triscape.mxs',
  6367. 'mxu' => 'video/vnd.mpegurl',
  6368. 'n-gage' => 'application/vnd.nokia.n-gage.symbian.install',
  6369. 'n3' => 'text/n3',
  6370. 'nb' => 'application/mathematica',
  6371. 'nbp' => 'application/vnd.wolfram.player',
  6372. 'nc' => 'application/x-netcdf',
  6373. 'ncx' => 'application/x-dtbncx+xml',
  6374. 'nfo' => 'text/x-nfo',
  6375. 'ngdat' => 'application/vnd.nokia.n-gage.data',
  6376. 'nitf' => 'application/vnd.nitf',
  6377. 'nlu' => 'application/vnd.neurolanguage.nlu',
  6378. 'nml' => 'application/vnd.enliven',
  6379. 'nnd' => 'application/vnd.noblenet-directory',
  6380. 'nns' => 'application/vnd.noblenet-sealer',
  6381. 'nnw' => 'application/vnd.noblenet-web',
  6382. 'npx' => 'image/vnd.net-fpx',
  6383. 'nsc' => 'application/x-conference',
  6384. 'nsf' => 'application/vnd.lotus-notes',
  6385. 'ntf' => 'application/vnd.nitf',
  6386. 'nzb' => 'application/x-nzb',
  6387. 'oa2' => 'application/vnd.fujitsu.oasys2',
  6388. 'oa3' => 'application/vnd.fujitsu.oasys3',
  6389. 'oas' => 'application/vnd.fujitsu.oasys',
  6390. 'obd' => 'application/x-msbinder',
  6391. 'obj' => 'application/x-tgif',
  6392. 'oda' => 'application/oda',
  6393. 'odb' => 'application/vnd.oasis.opendocument.database',
  6394. 'odc' => 'application/vnd.oasis.opendocument.chart',
  6395. 'odf' => 'application/vnd.oasis.opendocument.formula',
  6396. 'odft' => 'application/vnd.oasis.opendocument.formula-template',
  6397. 'odg' => 'application/vnd.oasis.opendocument.graphics',
  6398. 'odi' => 'application/vnd.oasis.opendocument.image',
  6399. 'odm' => 'application/vnd.oasis.opendocument.text-master',
  6400. 'odp' => 'application/vnd.oasis.opendocument.presentation',
  6401. 'ods' => 'application/vnd.oasis.opendocument.spreadsheet',
  6402. 'odt' => 'application/vnd.oasis.opendocument.text',
  6403. 'oga' => 'audio/ogg',
  6404. 'ogg' => 'audio/ogg',
  6405. 'ogv' => 'video/ogg',
  6406. 'ogx' => 'application/ogg',
  6407. 'omdoc' => 'application/omdoc+xml',
  6408. 'onepkg' => 'application/onenote',
  6409. 'onetmp' => 'application/onenote',
  6410. 'onetoc' => 'application/onenote',
  6411. 'onetoc2' => 'application/onenote',
  6412. 'opf' => 'application/oebps-package+xml',
  6413. 'opml' => 'text/x-opml',
  6414. 'oprc' => 'application/vnd.palm',
  6415. 'org' => 'application/vnd.lotus-organizer',
  6416. 'osf' => 'application/vnd.yamaha.openscoreformat',
  6417. 'osfpvg' => 'application/vnd.yamaha.openscoreformat.osfpvg+xml',
  6418. 'otc' => 'application/vnd.oasis.opendocument.chart-template',
  6419. 'otf' => 'application/x-font-otf',
  6420. 'otg' => 'application/vnd.oasis.opendocument.graphics-template',
  6421. 'oth' => 'application/vnd.oasis.opendocument.text-web',
  6422. 'oti' => 'application/vnd.oasis.opendocument.image-template',
  6423. 'otp' => 'application/vnd.oasis.opendocument.presentation-template',
  6424. 'ots' => 'application/vnd.oasis.opendocument.spreadsheet-template',
  6425. 'ott' => 'application/vnd.oasis.opendocument.text-template',
  6426. 'oxps' => 'application/oxps',
  6427. 'oxt' => 'application/vnd.openofficeorg.extension',
  6428. 'p' => 'text/x-pascal',
  6429. 'p10' => 'application/pkcs10',
  6430. 'p12' => 'application/x-pkcs12',
  6431. 'p7b' => 'application/x-pkcs7-certificates',
  6432. 'p7c' => 'application/pkcs7-mime',
  6433. 'p7m' => 'application/pkcs7-mime',
  6434. 'p7r' => 'application/x-pkcs7-certreqresp',
  6435. 'p7s' => 'application/pkcs7-signature',
  6436. 'p8' => 'application/pkcs8',
  6437. 'pas' => 'text/x-pascal',
  6438. 'paw' => 'application/vnd.pawaafile',
  6439. 'pbd' => 'application/vnd.powerbuilder6',
  6440. 'pbm' => 'image/x-portable-bitmap',
  6441. 'pcap' => 'application/vnd.tcpdump.pcap',
  6442. 'pcf' => 'application/x-font-pcf',
  6443. 'pcl' => 'application/vnd.hp-pcl',
  6444. 'pclxl' => 'application/vnd.hp-pclxl',
  6445. 'pct' => 'image/x-pict',
  6446. 'pcurl' => 'application/vnd.curl.pcurl',
  6447. 'pcx' => 'image/x-pcx',
  6448. 'pdb' => 'application/vnd.palm',
  6449. 'pdf' => 'application/pdf',
  6450. 'pfa' => 'application/x-font-type1',
  6451. 'pfb' => 'application/x-font-type1',
  6452. 'pfm' => 'application/x-font-type1',
  6453. 'pfr' => 'application/font-tdpfr',
  6454. 'pfx' => 'application/x-pkcs12',
  6455. 'pgm' => 'image/x-portable-graymap',
  6456. 'pgn' => 'application/x-chess-pgn',
  6457. 'pgp' => 'application/pgp-encrypted',
  6458. 'pic' => 'image/x-pict',
  6459. 'pkg' => 'application/octet-stream',
  6460. 'pki' => 'application/pkixcmp',
  6461. 'pkipath' => 'application/pkix-pkipath',
  6462. 'plb' => 'application/vnd.3gpp.pic-bw-large',
  6463. 'plc' => 'application/vnd.mobius.plc',
  6464. 'plf' => 'application/vnd.pocketlearn',
  6465. 'pls' => 'application/pls+xml',
  6466. 'pml' => 'application/vnd.ctc-posml',
  6467. 'png' => 'image/png',
  6468. 'pnm' => 'image/x-portable-anymap',
  6469. 'portpkg' => 'application/vnd.macports.portpkg',
  6470. 'pot' => 'application/vnd.ms-powerpoint',
  6471. 'potm' => 'application/vnd.ms-powerpoint.template.macroenabled.12',
  6472. 'potx' => 'application/vnd.openxmlformats-officedocument.presentationml.template',
  6473. 'ppam' => 'application/vnd.ms-powerpoint.addin.macroenabled.12',
  6474. 'ppd' => 'application/vnd.cups-ppd',
  6475. 'ppm' => 'image/x-portable-pixmap',
  6476. 'pps' => 'application/vnd.ms-powerpoint',
  6477. 'ppsm' => 'application/vnd.ms-powerpoint.slideshow.macroenabled.12',
  6478. 'ppsx' => 'application/vnd.openxmlformats-officedocument.presentationml.slideshow',
  6479. 'ppt' => 'application/vnd.ms-powerpoint',
  6480. 'pptm' => 'application/vnd.ms-powerpoint.presentation.macroenabled.12',
  6481. 'pptx' => 'application/vnd.openxmlformats-officedocument.presentationml.presentation',
  6482. 'pqa' => 'application/vnd.palm',
  6483. 'prc' => 'application/x-mobipocket-ebook',
  6484. 'pre' => 'application/vnd.lotus-freelance',
  6485. 'prf' => 'application/pics-rules',
  6486. 'ps' => 'application/postscript',
  6487. 'psb' => 'application/vnd.3gpp.pic-bw-small',
  6488. 'psd' => 'image/vnd.adobe.photoshop',
  6489. 'psf' => 'application/x-font-linux-psf',
  6490. 'pskcxml' => 'application/pskc+xml',
  6491. 'ptid' => 'application/vnd.pvi.ptid1',
  6492. 'pub' => 'application/x-mspublisher',
  6493. 'pvb' => 'application/vnd.3gpp.pic-bw-var',
  6494. 'pwn' => 'application/vnd.3m.post-it-notes',
  6495. 'pya' => 'audio/vnd.ms-playready.media.pya',
  6496. 'pyv' => 'video/vnd.ms-playready.media.pyv',
  6497. 'qam' => 'application/vnd.epson.quickanime',
  6498. 'qbo' => 'application/vnd.intu.qbo',
  6499. 'qfx' => 'application/vnd.intu.qfx',
  6500. 'qps' => 'application/vnd.publishare-delta-tree',
  6501. 'qt' => 'video/quicktime',
  6502. 'qwd' => 'application/vnd.quark.quarkxpress',
  6503. 'qwt' => 'application/vnd.quark.quarkxpress',
  6504. 'qxb' => 'application/vnd.quark.quarkxpress',
  6505. 'qxd' => 'application/vnd.quark.quarkxpress',
  6506. 'qxl' => 'application/vnd.quark.quarkxpress',
  6507. 'qxt' => 'application/vnd.quark.quarkxpress',
  6508. 'ra' => 'audio/x-pn-realaudio',
  6509. 'ram' => 'audio/x-pn-realaudio',
  6510. 'rar' => 'application/x-rar-compressed',
  6511. 'ras' => 'image/x-cmu-raster',
  6512. 'rcprofile' => 'application/vnd.ipunplugged.rcprofile',
  6513. 'rdf' => 'application/rdf+xml',
  6514. 'rdz' => 'application/vnd.data-vision.rdz',
  6515. 'rep' => 'application/vnd.businessobjects',
  6516. 'res' => 'application/x-dtbresource+xml',
  6517. 'rgb' => 'image/x-rgb',
  6518. 'rif' => 'application/reginfo+xml',
  6519. 'rip' => 'audio/vnd.rip',
  6520. 'ris' => 'application/x-research-info-systems',
  6521. 'rl' => 'application/resource-lists+xml',
  6522. 'rlc' => 'image/vnd.fujixerox.edmics-rlc',
  6523. 'rld' => 'application/resource-lists-diff+xml',
  6524. 'rm' => 'application/vnd.rn-realmedia',
  6525. 'rmi' => 'audio/midi',
  6526. 'rmp' => 'audio/x-pn-realaudio-plugin',
  6527. 'rms' => 'application/vnd.jcp.javame.midlet-rms',
  6528. 'rmvb' => 'application/vnd.rn-realmedia-vbr',
  6529. 'rnc' => 'application/relax-ng-compact-syntax',
  6530. 'roa' => 'application/rpki-roa',
  6531. 'roff' => 'text/troff',
  6532. 'rp9' => 'application/vnd.cloanto.rp9',
  6533. 'rpss' => 'application/vnd.nokia.radio-presets',
  6534. 'rpst' => 'application/vnd.nokia.radio-preset',
  6535. 'rq' => 'application/sparql-query',
  6536. 'rs' => 'application/rls-services+xml',
  6537. 'rsd' => 'application/rsd+xml',
  6538. 'rss' => 'application/rss+xml',
  6539. 'rtf' => 'application/rtf',
  6540. 'rtx' => 'text/richtext',
  6541. 's' => 'text/x-asm',
  6542. 's3m' => 'audio/s3m',
  6543. 'saf' => 'application/vnd.yamaha.smaf-audio',
  6544. 'sbml' => 'application/sbml+xml',
  6545. 'sc' => 'application/vnd.ibm.secure-container',
  6546. 'scd' => 'application/x-msschedule',
  6547. 'scm' => 'application/vnd.lotus-screencam',
  6548. 'scq' => 'application/scvp-cv-request',
  6549. 'scs' => 'application/scvp-cv-response',
  6550. 'scurl' => 'text/vnd.curl.scurl',
  6551. 'sda' => 'application/vnd.stardivision.draw',
  6552. 'sdc' => 'application/vnd.stardivision.calc',
  6553. 'sdd' => 'application/vnd.stardivision.impress',
  6554. 'sdkd' => 'application/vnd.solent.sdkm+xml',
  6555. 'sdkm' => 'application/vnd.solent.sdkm+xml',
  6556. 'sdp' => 'application/sdp',
  6557. 'sdw' => 'application/vnd.stardivision.writer',
  6558. 'see' => 'application/vnd.seemail',
  6559. 'seed' => 'application/vnd.fdsn.seed',
  6560. 'sema' => 'application/vnd.sema',
  6561. 'semd' => 'application/vnd.semd',
  6562. 'semf' => 'application/vnd.semf',
  6563. 'ser' => 'application/java-serialized-object',
  6564. 'setpay' => 'application/set-payment-initiation',
  6565. 'setreg' => 'application/set-registration-initiation',
  6566. 'sfd-hdstx' => 'application/vnd.hydrostatix.sof-data',
  6567. 'sfs' => 'application/vnd.spotfire.sfs',
  6568. 'sfv' => 'text/x-sfv',
  6569. 'sgi' => 'image/sgi',
  6570. 'sgl' => 'application/vnd.stardivision.writer-global',
  6571. 'sgm' => 'text/sgml',
  6572. 'sgml' => 'text/sgml',
  6573. 'sh' => 'application/x-sh',
  6574. 'shar' => 'application/x-shar',
  6575. 'shf' => 'application/shf+xml',
  6576. 'sid' => 'image/x-mrsid-image',
  6577. 'sig' => 'application/pgp-signature',
  6578. 'sil' => 'audio/silk',
  6579. 'silo' => 'model/mesh',
  6580. 'sis' => 'application/vnd.symbian.install',
  6581. 'sisx' => 'application/vnd.symbian.install',
  6582. 'sit' => 'application/x-stuffit',
  6583. 'sitx' => 'application/x-stuffitx',
  6584. 'skd' => 'application/vnd.koan',
  6585. 'skm' => 'application/vnd.koan',
  6586. 'skp' => 'application/vnd.koan',
  6587. 'skt' => 'application/vnd.koan',
  6588. 'sldm' => 'application/vnd.ms-powerpoint.slide.macroenabled.12',
  6589. 'sldx' => 'application/vnd.openxmlformats-officedocument.presentationml.slide',
  6590. 'slt' => 'application/vnd.epson.salt',
  6591. 'sm' => 'application/vnd.stepmania.stepchart',
  6592. 'smf' => 'application/vnd.stardivision.math',
  6593. 'smi' => 'application/smil+xml',
  6594. 'smil' => 'application/smil+xml',
  6595. 'smv' => 'video/x-smv',
  6596. 'smzip' => 'application/vnd.stepmania.package',
  6597. 'snd' => 'audio/basic',
  6598. 'snf' => 'application/x-font-snf',
  6599. 'so' => 'application/octet-stream',
  6600. 'spc' => 'application/x-pkcs7-certificates',
  6601. 'spf' => 'application/vnd.yamaha.smaf-phrase',
  6602. 'spl' => 'application/x-futuresplash',
  6603. 'spot' => 'text/vnd.in3d.spot',
  6604. 'spp' => 'application/scvp-vp-response',
  6605. 'spq' => 'application/scvp-vp-request',
  6606. 'spx' => 'audio/ogg',
  6607. 'sql' => 'application/x-sql',
  6608. 'src' => 'application/x-wais-source',
  6609. 'srt' => 'application/x-subrip',
  6610. 'sru' => 'application/sru+xml',
  6611. 'srx' => 'application/sparql-results+xml',
  6612. 'ssdl' => 'application/ssdl+xml',
  6613. 'sse' => 'application/vnd.kodak-descriptor',
  6614. 'ssf' => 'application/vnd.epson.ssf',
  6615. 'ssml' => 'application/ssml+xml',
  6616. 'st' => 'application/vnd.sailingtracker.track',
  6617. 'stc' => 'application/vnd.sun.xml.calc.template',
  6618. 'std' => 'application/vnd.sun.xml.draw.template',
  6619. 'stf' => 'application/vnd.wt.stf',
  6620. 'sti' => 'application/vnd.sun.xml.impress.template',
  6621. 'stk' => 'application/hyperstudio',
  6622. 'stl' => 'application/vnd.ms-pki.stl',
  6623. 'str' => 'application/vnd.pg.format',
  6624. 'stw' => 'application/vnd.sun.xml.writer.template',
  6625. 'sub' => 'image/vnd.dvb.subtitle',
  6626. 'sub' => 'text/vnd.dvb.subtitle',
  6627. 'sus' => 'application/vnd.sus-calendar',
  6628. 'susp' => 'application/vnd.sus-calendar',
  6629. 'sv4cpio' => 'application/x-sv4cpio',
  6630. 'sv4crc' => 'application/x-sv4crc',
  6631. 'svc' => 'application/vnd.dvb.service',
  6632. 'svd' => 'application/vnd.svd',
  6633. 'svg' => 'image/svg+xml',
  6634. 'svgz' => 'image/svg+xml',
  6635. 'swa' => 'application/x-director',
  6636. 'swf' => 'application/x-shockwave-flash',
  6637. 'swi' => 'application/vnd.aristanetworks.swi',
  6638. 'sxc' => 'application/vnd.sun.xml.calc',
  6639. 'sxd' => 'application/vnd.sun.xml.draw',
  6640. 'sxg' => 'application/vnd.sun.xml.writer.global',
  6641. 'sxi' => 'application/vnd.sun.xml.impress',
  6642. 'sxm' => 'application/vnd.sun.xml.math',
  6643. 'sxw' => 'application/vnd.sun.xml.writer',
  6644. 't' => 'text/troff',
  6645. 't3' => 'application/x-t3vm-image',
  6646. 'taglet' => 'application/vnd.mynfc',
  6647. 'tao' => 'application/vnd.tao.intent-module-archive',
  6648. 'tar' => 'application/x-tar',
  6649. 'tcap' => 'application/vnd.3gpp2.tcap',
  6650. 'tcl' => 'application/x-tcl',
  6651. 'teacher' => 'application/vnd.smart.teacher',
  6652. 'tei' => 'application/tei+xml',
  6653. 'teicorpus' => 'application/tei+xml',
  6654. 'tex' => 'application/x-tex',
  6655. 'texi' => 'application/x-texinfo',
  6656. 'texinfo' => 'application/x-texinfo',
  6657. 'text' => 'text/plain',
  6658. 'tfi' => 'application/thraud+xml',
  6659. 'tfm' => 'application/x-tex-tfm',
  6660. 'tga' => 'image/x-tga',
  6661. 'thmx' => 'application/vnd.ms-officetheme',
  6662. 'tif' => 'image/tiff',
  6663. 'tiff' => 'image/tiff',
  6664. 'tmo' => 'application/vnd.tmobile-livetv',
  6665. 'torrent' => 'application/x-bittorrent',
  6666. 'tpl' => 'application/vnd.groove-tool-template',
  6667. 'tpt' => 'application/vnd.trid.tpt',
  6668. 'tr' => 'text/troff',
  6669. 'tra' => 'application/vnd.trueapp',
  6670. 'trm' => 'application/x-msterminal',
  6671. 'tsd' => 'application/timestamped-data',
  6672. 'tsv' => 'text/tab-separated-values',
  6673. 'ttc' => 'application/x-font-ttf',
  6674. 'ttf' => 'application/x-font-ttf',
  6675. 'ttl' => 'text/turtle',
  6676. 'twd' => 'application/vnd.simtech-mindmapper',
  6677. 'twds' => 'application/vnd.simtech-mindmapper',
  6678. 'txd' => 'application/vnd.genomatix.tuxedo',
  6679. 'txf' => 'application/vnd.mobius.txf',
  6680. 'txt' => 'text/plain',
  6681. 'u32' => 'application/x-authorware-bin',
  6682. 'udeb' => 'application/x-debian-package',
  6683. 'ufd' => 'application/vnd.ufdl',
  6684. 'ufdl' => 'application/vnd.ufdl',
  6685. 'ulx' => 'application/x-glulx',
  6686. 'umj' => 'application/vnd.umajin',
  6687. 'unityweb' => 'application/vnd.unity',
  6688. 'uoml' => 'application/vnd.uoml+xml',
  6689. 'uri' => 'text/uri-list',
  6690. 'uris' => 'text/uri-list',
  6691. 'urls' => 'text/uri-list',
  6692. 'ustar' => 'application/x-ustar',
  6693. 'utz' => 'application/vnd.uiq.theme',
  6694. 'uu' => 'text/x-uuencode',
  6695. 'uva' => 'audio/vnd.dece.audio',
  6696. 'uvd' => 'application/vnd.dece.data',
  6697. 'uvf' => 'application/vnd.dece.data',
  6698. 'uvg' => 'image/vnd.dece.graphic',
  6699. 'uvh' => 'video/vnd.dece.hd',
  6700. 'uvi' => 'image/vnd.dece.graphic',
  6701. 'uvm' => 'video/vnd.dece.mobile',
  6702. 'uvp' => 'video/vnd.dece.pd',
  6703. 'uvs' => 'video/vnd.dece.sd',
  6704. 'uvt' => 'application/vnd.dece.ttml+xml',
  6705. 'uvu' => 'video/vnd.uvvu.mp4',
  6706. 'uvv' => 'video/vnd.dece.video',
  6707. 'uvva' => 'audio/vnd.dece.audio',
  6708. 'uvvd' => 'application/vnd.dece.data',
  6709. 'uvvf' => 'application/vnd.dece.data',
  6710. 'uvvg' => 'image/vnd.dece.graphic',
  6711. 'uvvh' => 'video/vnd.dece.hd',
  6712. 'uvvi' => 'image/vnd.dece.graphic',
  6713. 'uvvm' => 'video/vnd.dece.mobile',
  6714. 'uvvp' => 'video/vnd.dece.pd',
  6715. 'uvvs' => 'video/vnd.dece.sd',
  6716. 'uvvt' => 'application/vnd.dece.ttml+xml',
  6717. 'uvvu' => 'video/vnd.uvvu.mp4',
  6718. 'uvvv' => 'video/vnd.dece.video',
  6719. 'uvvx' => 'application/vnd.dece.unspecified',
  6720. 'uvvz' => 'application/vnd.dece.zip',
  6721. 'uvx' => 'application/vnd.dece.unspecified',
  6722. 'uvz' => 'application/vnd.dece.zip',
  6723. 'vcard' => 'text/vcard',
  6724. 'vcd' => 'application/x-cdlink',
  6725. 'vcf' => 'text/x-vcard',
  6726. 'vcg' => 'application/vnd.groove-vcard',
  6727. 'vcs' => 'text/x-vcalendar',
  6728. 'vcx' => 'application/vnd.vcx',
  6729. 'vis' => 'application/vnd.visionary',
  6730. 'viv' => 'video/vnd.vivo',
  6731. 'vob' => 'video/x-ms-vob',
  6732. 'vor' => 'application/vnd.stardivision.writer',
  6733. 'vox' => 'application/x-authorware-bin',
  6734. 'vrml' => 'model/vrml',
  6735. 'vsd' => 'application/vnd.visio',
  6736. 'vsf' => 'application/vnd.vsf',
  6737. 'vss' => 'application/vnd.visio',
  6738. 'vst' => 'application/vnd.visio',
  6739. 'vsw' => 'application/vnd.visio',
  6740. 'vtu' => 'model/vnd.vtu',
  6741. 'vxml' => 'application/voicexml+xml',
  6742. 'w3d' => 'application/x-director',
  6743. 'wad' => 'application/x-doom',
  6744. 'wav' => 'audio/x-wav',
  6745. 'wax' => 'audio/x-ms-wax',
  6746. 'wbmp' => 'image/vnd.wap.wbmp',
  6747. 'wbs' => 'application/vnd.criticaltools.wbs+xml',
  6748. 'wbxml' => 'application/vnd.wap.wbxml',
  6749. 'wcm' => 'application/vnd.ms-works',
  6750. 'wdb' => 'application/vnd.ms-works',
  6751. 'wdp' => 'image/vnd.ms-photo',
  6752. 'weba' => 'audio/webm',
  6753. 'webm' => 'video/webm',
  6754. 'webp' => 'image/webp',
  6755. 'wg' => 'application/vnd.pmi.widget',
  6756. 'wgt' => 'application/widget',
  6757. 'wks' => 'application/vnd.ms-works',
  6758. 'wm' => 'video/x-ms-wm',
  6759. 'wma' => 'audio/x-ms-wma',
  6760. 'wmd' => 'application/x-ms-wmd',
  6761. 'wmf' => 'application/x-msmetafile',
  6762. 'wml' => 'text/vnd.wap.wml',
  6763. 'wmlc' => 'application/vnd.wap.wmlc',
  6764. 'wmls' => 'text/vnd.wap.wmlscript',
  6765. 'wmlsc' => 'application/vnd.wap.wmlscriptc',
  6766. 'wmv' => 'video/x-ms-wmv',
  6767. 'wmx' => 'video/x-ms-wmx',
  6768. 'wmz' => 'application/x-ms-wmz',
  6769. 'wmz' => 'application/x-msmetafile',
  6770. 'woff' => 'application/font-woff',
  6771. 'wpd' => 'application/vnd.wordperfect',
  6772. 'wpl' => 'application/vnd.ms-wpl',
  6773. 'wps' => 'application/vnd.ms-works',
  6774. 'wqd' => 'application/vnd.wqd',
  6775. 'wri' => 'application/x-mswrite',
  6776. 'wrl' => 'model/vrml',
  6777. 'wsdl' => 'application/wsdl+xml',
  6778. 'wspolicy' => 'application/wspolicy+xml',
  6779. 'wtb' => 'application/vnd.webturbo',
  6780. 'wvx' => 'video/x-ms-wvx',
  6781. 'x32' => 'application/x-authorware-bin',
  6782. 'x3d' => 'model/x3d+xml',
  6783. 'x3db' => 'model/x3d+binary',
  6784. 'x3dbz' => 'model/x3d+binary',
  6785. 'x3dv' => 'model/x3d+vrml',
  6786. 'x3dvz' => 'model/x3d+vrml',
  6787. 'x3dz' => 'model/x3d+xml',
  6788. 'xaml' => 'application/xaml+xml',
  6789. 'xap' => 'application/x-silverlight-app',
  6790. 'xar' => 'application/vnd.xara',
  6791. 'xbap' => 'application/x-ms-xbap',
  6792. 'xbd' => 'application/vnd.fujixerox.docuworks.binder',
  6793. 'xbm' => 'image/x-xbitmap',
  6794. 'xdf' => 'application/xcap-diff+xml',
  6795. 'xdm' => 'application/vnd.syncml.dm+xml',
  6796. 'xdp' => 'application/vnd.adobe.xdp+xml',
  6797. 'xdssc' => 'application/dssc+xml',
  6798. 'xdw' => 'application/vnd.fujixerox.docuworks',
  6799. 'xenc' => 'application/xenc+xml',
  6800. 'xer' => 'application/patch-ops-error+xml',
  6801. 'xfdf' => 'application/vnd.adobe.xfdf',
  6802. 'xfdl' => 'application/vnd.xfdl',
  6803. 'xht' => 'application/xhtml+xml',
  6804. 'xhtml' => 'application/xhtml+xml',
  6805. 'xhvml' => 'application/xv+xml',
  6806. 'xif' => 'image/vnd.xiff',
  6807. 'xla' => 'application/vnd.ms-excel',
  6808. 'xlam' => 'application/vnd.ms-excel.addin.macroenabled.12',
  6809. 'xlc' => 'application/vnd.ms-excel',
  6810. 'xlf' => 'application/x-xliff+xml',
  6811. 'xlm' => 'application/vnd.ms-excel',
  6812. 'xls' => 'application/vnd.ms-excel',
  6813. 'xlsb' => 'application/vnd.ms-excel.sheet.binary.macroenabled.12',
  6814. 'xlsm' => 'application/vnd.ms-excel.sheet.macroenabled.12',
  6815. 'xlsx' => 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
  6816. 'xlt' => 'application/vnd.ms-excel',
  6817. 'xltm' => 'application/vnd.ms-excel.template.macroenabled.12',
  6818. 'xltx' => 'application/vnd.openxmlformats-officedocument.spreadsheetml.template',
  6819. 'xlw' => 'application/vnd.ms-excel',
  6820. 'xm' => 'audio/xm',
  6821. 'xml' => 'application/xml',
  6822. 'xo' => 'application/vnd.olpc-sugar',
  6823. 'xop' => 'application/xop+xml',
  6824. 'xpi' => 'application/x-xpinstall',
  6825. 'xpl' => 'application/xproc+xml',
  6826. 'xpm' => 'image/x-xpixmap',
  6827. 'xpr' => 'application/vnd.is-xpr',
  6828. 'xps' => 'application/vnd.ms-xpsdocument',
  6829. 'xpw' => 'application/vnd.intercon.formnet',
  6830. 'xpx' => 'application/vnd.intercon.formnet',
  6831. 'xsl' => 'application/xml',
  6832. 'xslt' => 'application/xslt+xml',
  6833. 'xsm' => 'application/vnd.syncml+xml',
  6834. 'xspf' => 'application/xspf+xml',
  6835. 'xul' => 'application/vnd.mozilla.xul+xml',
  6836. 'xvm' => 'application/xv+xml',
  6837. 'xvml' => 'application/xv+xml',
  6838. 'xwd' => 'image/x-xwindowdump',
  6839. 'xyz' => 'chemical/x-xyz',
  6840. 'xz' => 'application/x-xz',
  6841. 'yang' => 'application/yang',
  6842. 'yin' => 'application/yin+xml',
  6843. 'z1' => 'application/x-zmachine',
  6844. 'z2' => 'application/x-zmachine',
  6845. 'z3' => 'application/x-zmachine',
  6846. 'z4' => 'application/x-zmachine',
  6847. 'z5' => 'application/x-zmachine',
  6848. 'z6' => 'application/x-zmachine',
  6849. 'z7' => 'application/x-zmachine',
  6850. 'z8' => 'application/x-zmachine',
  6851. 'zaz' => 'application/vnd.zzazz.deck+xml',
  6852. 'zip' => 'application/zip',
  6853. 'zir' => 'application/vnd.zul',
  6854. 'zirz' => 'application/vnd.zul',
  6855. 'zmm' => 'application/vnd.handheld-entertainment+xml'
  6856. );
  6857. }
  6858. // Always run this
  6859. dependCheck();