api.lib.php 303 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396539753985399540054015402540354045405540654075408540954105411541254135414541554165417541854195420542154225423542454255426542754285429543054315432543354345435543654375438543954405441544254435444544554465447544854495450545154525453545454555456545754585459546054615462546354645465546654675468546954705471547254735474547554765477547854795480548154825483548454855486548754885489549054915492549354945495549654975498549955005501550255035504550555065507550855095510551155125513551455155516551755185519552055215522552355245525552655275528552955305531553255335534553555365537553855395540554155425543554455455546554755485549555055515552555355545555555655575558555955605561556255635564556555665567556855695570557155725573557455755576557755785579558055815582558355845585558655875588558955905591559255935594559555965597559855995600560156025603560456055606560756085609561056115612561356145615561656175618561956205621562256235624562556265627562856295630563156325633563456355636563756385639564056415642564356445645564656475648564956505651565256535654565556565657565856595660566156625663566456655666566756685669567056715672567356745675567656775678567956805681568256835684568556865687568856895690569156925693569456955696569756985699570057015702570357045705570657075708570957105711571257135714571557165717571857195720572157225723572457255726572757285729573057315732573357345735573657375738573957405741574257435744574557465747574857495750575157525753575457555756575757585759576057615762576357645765576657675768576957705771577257735774577557765777577857795780578157825783578457855786578757885789579057915792579357945795579657975798579958005801580258035804580558065807580858095810581158125813581458155816581758185819582058215822582358245825582658275828582958305831583258335834583558365837583858395840584158425843584458455846584758485849585058515852585358545855585658575858585958605861586258635864586558665867586858695870587158725873587458755876587758785879588058815882588358845885588658875888588958905891589258935894589558965897589858995900590159025903590459055906590759085909591059115912591359145915591659175918591959205921592259235924592559265927592859295930593159325933593459355936593759385939594059415942594359445945594659475948594959505951595259535954595559565957595859595960596159625963596459655966596759685969597059715972597359745975597659775978597959805981598259835984598559865987598859895990599159925993599459955996599759985999600060016002600360046005600660076008600960106011601260136014601560166017601860196020602160226023602460256026602760286029603060316032603360346035603660376038603960406041604260436044604560466047604860496050605160526053605460556056605760586059606060616062606360646065606660676068606960706071607260736074607560766077607860796080608160826083608460856086608760886089609060916092609360946095609660976098609961006101610261036104610561066107610861096110611161126113611461156116611761186119612061216122612361246125612661276128612961306131613261336134613561366137613861396140614161426143614461456146614761486149615061516152615361546155615661576158615961606161616261636164616561666167616861696170617161726173617461756176617761786179618061816182618361846185618661876188618961906191619261936194619561966197619861996200620162026203620462056206620762086209621062116212621362146215621662176218621962206221622262236224622562266227622862296230623162326233623462356236623762386239624062416242624362446245624662476248624962506251625262536254625562566257625862596260626162626263626462656266626762686269627062716272627362746275627662776278627962806281628262836284628562866287628862896290629162926293629462956296629762986299630063016302630363046305630663076308630963106311631263136314631563166317631863196320632163226323632463256326632763286329633063316332633363346335633663376338633963406341634263436344634563466347634863496350635163526353635463556356635763586359636063616362636363646365636663676368636963706371637263736374637563766377637863796380638163826383638463856386638763886389639063916392639363946395639663976398639964006401640264036404640564066407640864096410641164126413641464156416641764186419642064216422642364246425642664276428642964306431643264336434643564366437643864396440644164426443644464456446644764486449645064516452645364546455645664576458645964606461646264636464646564666467646864696470647164726473647464756476647764786479648064816482648364846485648664876488648964906491649264936494649564966497649864996500650165026503650465056506650765086509651065116512651365146515651665176518651965206521652265236524652565266527652865296530653165326533653465356536653765386539654065416542654365446545654665476548654965506551655265536554655565566557655865596560656165626563656465656566656765686569657065716572657365746575657665776578657965806581658265836584658565866587658865896590659165926593659465956596659765986599660066016602660366046605660666076608660966106611661266136614661566166617661866196620662166226623662466256626662766286629663066316632663366346635663666376638663966406641664266436644664566466647664866496650665166526653665466556656665766586659666066616662666366646665666666676668666966706671667266736674667566766677667866796680668166826683668466856686668766886689669066916692669366946695669666976698669967006701670267036704670567066707670867096710671167126713671467156716671767186719672067216722672367246725672667276728672967306731673267336734673567366737673867396740674167426743674467456746674767486749675067516752675367546755675667576758675967606761676267636764676567666767676867696770677167726773677467756776677767786779678067816782678367846785678667876788678967906791679267936794679567966797679867996800680168026803680468056806680768086809681068116812681368146815681668176818681968206821682268236824682568266827682868296830683168326833683468356836683768386839684068416842684368446845684668476848684968506851685268536854685568566857685868596860686168626863686468656866686768686869687068716872687368746875687668776878687968806881688268836884688568866887688868896890689168926893689468956896689768986899690069016902690369046905690669076908690969106911691269136914691569166917691869196920692169226923692469256926692769286929693069316932693369346935693669376938693969406941694269436944694569466947694869496950695169526953695469556956695769586959696069616962696369646965696669676968696969706971697269736974697569766977697869796980698169826983698469856986698769886989699069916992699369946995699669976998699970007001700270037004700570067007700870097010701170127013701470157016701770187019702070217022702370247025702670277028702970307031703270337034703570367037703870397040704170427043704470457046704770487049705070517052705370547055705670577058705970607061706270637064706570667067706870697070707170727073707470757076707770787079708070817082708370847085708670877088708970907091709270937094709570967097709870997100710171027103710471057106710771087109711071117112711371147115711671177118711971207121712271237124712571267127712871297130713171327133713471357136713771387139714071417142714371447145714671477148714971507151715271537154715571567157715871597160716171627163716471657166716771687169717071717172717371747175717671777178717971807181718271837184718571867187718871897190719171927193719471957196719771987199720072017202720372047205720672077208720972107211721272137214721572167217721872197220722172227223722472257226722772287229723072317232723372347235723672377238723972407241724272437244724572467247724872497250725172527253725472557256725772587259726072617262726372647265726672677268726972707271727272737274727572767277727872797280728172827283728472857286728772887289729072917292729372947295729672977298729973007301730273037304730573067307730873097310731173127313731473157316731773187319732073217322732373247325732673277328732973307331733273337334733573367337733873397340734173427343734473457346734773487349735073517352735373547355735673577358735973607361736273637364736573667367736873697370737173727373737473757376737773787379738073817382738373847385738673877388738973907391739273937394739573967397739873997400740174027403740474057406740774087409741074117412741374147415741674177418741974207421742274237424742574267427742874297430743174327433743474357436743774387439744074417442744374447445744674477448744974507451745274537454745574567457745874597460746174627463746474657466746774687469747074717472747374747475747674777478747974807481748274837484748574867487748874897490749174927493749474957496749774987499750075017502750375047505750675077508750975107511751275137514751575167517751875197520752175227523752475257526752775287529753075317532753375347535753675377538753975407541754275437544754575467547754875497550755175527553755475557556755775587559756075617562756375647565756675677568756975707571757275737574757575767577757875797580758175827583758475857586758775887589759075917592759375947595759675977598759976007601760276037604760576067607760876097610761176127613761476157616761776187619762076217622762376247625762676277628762976307631763276337634763576367637763876397640764176427643764476457646764776487649765076517652765376547655765676577658765976607661766276637664766576667667766876697670767176727673767476757676767776787679768076817682768376847685768676877688768976907691769276937694769576967697769876997700770177027703770477057706770777087709771077117712771377147715771677177718771977207721772277237724772577267727772877297730773177327733773477357736773777387739774077417742774377447745774677477748774977507751775277537754775577567757775877597760776177627763776477657766776777687769777077717772777377747775777677777778777977807781778277837784778577867787778877897790779177927793779477957796779777987799780078017802780378047805780678077808780978107811781278137814781578167817781878197820782178227823782478257826782778287829783078317832783378347835783678377838783978407841784278437844784578467847784878497850785178527853785478557856785778587859786078617862786378647865786678677868786978707871787278737874787578767877787878797880788178827883788478857886788778887889789078917892789378947895789678977898789979007901790279037904790579067907790879097910791179127913791479157916791779187919792079217922792379247925792679277928792979307931793279337934793579367937793879397940794179427943794479457946794779487949795079517952795379547955795679577958795979607961796279637964796579667967796879697970797179727973797479757976797779787979798079817982798379847985798679877988798979907991799279937994799579967997799879998000800180028003800480058006800780088009801080118012801380148015801680178018801980208021802280238024802580268027802880298030803180328033803480358036803780388039804080418042804380448045804680478048804980508051805280538054805580568057805880598060806180628063806480658066806780688069807080718072807380748075807680778078807980808081808280838084808580868087808880898090809180928093809480958096809780988099810081018102810381048105810681078108810981108111811281138114811581168117811881198120812181228123812481258126812781288129813081318132813381348135813681378138813981408141814281438144814581468147814881498150815181528153815481558156815781588159816081618162816381648165816681678168816981708171817281738174817581768177817881798180818181828183818481858186818781888189819081918192819381948195819681978198819982008201820282038204820582068207820882098210821182128213821482158216821782188219822082218222822382248225822682278228822982308231823282338234823582368237823882398240824182428243824482458246824782488249825082518252825382548255825682578258825982608261826282638264826582668267826882698270827182728273827482758276827782788279828082818282828382848285828682878288828982908291829282938294829582968297829882998300830183028303830483058306830783088309831083118312831383148315831683178318831983208321832283238324832583268327832883298330833183328333833483358336833783388339834083418342834383448345834683478348834983508351835283538354835583568357835883598360836183628363836483658366836783688369837083718372837383748375837683778378837983808381838283838384838583868387838883898390839183928393839483958396839783988399840084018402840384048405840684078408840984108411841284138414841584168417841884198420842184228423842484258426842784288429843084318432843384348435843684378438843984408441844284438444844584468447844884498450845184528453845484558456845784588459846084618462846384648465846684678468846984708471847284738474847584768477847884798480848184828483848484858486848784888489849084918492849384948495849684978498849985008501850285038504850585068507850885098510851185128513851485158516851785188519852085218522852385248525852685278528852985308531853285338534853585368537853885398540854185428543854485458546854785488549855085518552855385548555855685578558855985608561856285638564856585668567856885698570857185728573857485758576857785788579858085818582858385848585858685878588858985908591859285938594859585968597859885998600860186028603860486058606860786088609861086118612861386148615861686178618861986208621862286238624862586268627862886298630863186328633863486358636863786388639864086418642864386448645864686478648864986508651865286538654865586568657865886598660866186628663866486658666866786688669867086718672867386748675867686778678867986808681868286838684868586868687868886898690869186928693869486958696869786988699870087018702870387048705870687078708870987108711871287138714871587168717871887198720872187228723872487258726872787288729873087318732873387348735873687378738873987408741874287438744874587468747874887498750875187528753875487558756875787588759876087618762876387648765876687678768876987708771877287738774877587768777877887798780878187828783878487858786878787888789879087918792879387948795879687978798879988008801880288038804880588068807880888098810881188128813881488158816881788188819882088218822882388248825882688278828882988308831883288338834883588368837883888398840884188428843884488458846884788488849885088518852885388548855885688578858885988608861886288638864886588668867886888698870887188728873887488758876887788788879888088818882888388848885888688878888888988908891889288938894889588968897889888998900890189028903890489058906890789088909891089118912891389148915891689178918891989208921892289238924892589268927892889298930893189328933893489358936893789388939894089418942894389448945894689478948894989508951895289538954895589568957895889598960896189628963896489658966896789688969897089718972897389748975897689778978897989808981898289838984898589868987898889898990899189928993899489958996899789988999900090019002900390049005900690079008900990109011901290139014901590169017901890199020902190229023902490259026902790289029903090319032903390349035903690379038903990409041904290439044904590469047904890499050905190529053905490559056905790589059906090619062906390649065906690679068906990709071907290739074907590769077907890799080908190829083908490859086908790889089909090919092909390949095909690979098909991009101910291039104910591069107910891099110911191129113911491159116911791189119912091219122912391249125912691279128912991309131913291339134913591369137913891399140914191429143914491459146914791489149915091519152915391549155915691579158915991609161916291639164916591669167916891699170917191729173917491759176917791789179918091819182918391849185918691879188918991909191919291939194919591969197919891999200920192029203920492059206920792089209921092119212921392149215921692179218921992209221922292239224922592269227922892299230923192329233923492359236923792389239924092419242924392449245924692479248924992509251925292539254925592569257925892599260926192629263926492659266926792689269927092719272927392749275927692779278927992809281928292839284928592869287928892899290929192929293929492959296929792989299930093019302930393049305930693079308930993109311931293139314931593169317931893199320932193229323932493259326932793289329933093319332933393349335933693379338933993409341934293439344934593469347934893499350935193529353935493559356935793589359936093619362936393649365936693679368936993709371937293739374937593769377937893799380938193829383938493859386938793889389939093919392939393949395939693979398939994009401940294039404940594069407940894099410941194129413941494159416941794189419942094219422942394249425942694279428942994309431943294339434943594369437943894399440944194429443944494459446944794489449945094519452945394549455945694579458945994609461946294639464946594669467946894699470947194729473947494759476947794789479948094819482948394849485948694879488948994909491949294939494949594969497949894999500950195029503950495059506950795089509951095119512951395149515951695179518951995209521952295239524952595269527952895299530953195329533953495359536953795389539954095419542954395449545954695479548954995509551955295539554955595569557955895599560956195629563956495659566956795689569957095719572957395749575957695779578957995809581958295839584958595869587958895899590959195929593959495959596959795989599960096019602
  1. <?php
  2. /* For licensing terms, see /license.txt */
  3. use Chamilo\CoreBundle\Entity\SettingsCurrent;
  4. use Chamilo\CourseBundle\Entity\CItemProperty;
  5. use Chamilo\UserBundle\Entity\User;
  6. use ChamiloSession as Session;
  7. use Symfony\Component\Finder\Finder;
  8. /**
  9. * This is a code library for Chamilo.
  10. * It is included by default in every Chamilo file (through including the global.inc.php)
  11. * This library is in process of being transferred to src/Chamilo/CoreBundle/Component/Utils/ChamiloApi.
  12. * Whenever a function is transferred to the ChamiloApi class, the places where it is used should include
  13. * the "use Chamilo\CoreBundle\Component\Utils\ChamiloApi;" statement.
  14. *
  15. * @package chamilo.library
  16. */
  17. // PHP version requirement.
  18. define('REQUIRED_PHP_VERSION', '5.6');
  19. define('REQUIRED_MIN_MEMORY_LIMIT', '128');
  20. define('REQUIRED_MIN_UPLOAD_MAX_FILESIZE', '10');
  21. define('REQUIRED_MIN_POST_MAX_SIZE', '10');
  22. // USER STATUS CONSTANTS
  23. /** global status of a user: student */
  24. define('STUDENT', 5);
  25. /** global status of a user: course manager */
  26. define('COURSEMANAGER', 1);
  27. /** global status of a user: session admin */
  28. define('SESSIONADMIN', 3);
  29. /** global status of a user: human ressource manager */
  30. define('DRH', 4);
  31. /** global status of a user: human ressource manager */
  32. define('ANONYMOUS', 6);
  33. /** global status of a user: low security, necessary for inserting data from
  34. * the teacher through HTMLPurifier */
  35. define('COURSEMANAGERLOWSECURITY', 10);
  36. // Soft user status
  37. define('PLATFORM_ADMIN', 11);
  38. define('SESSION_COURSE_COACH', 12);
  39. define('SESSION_GENERAL_COACH', 13);
  40. define('COURSE_STUDENT', 14); //student subscribed in a course
  41. define('SESSION_STUDENT', 15); //student subscribed in a session course
  42. define('COURSE_TUTOR', 16); // student is tutor of a course (NOT in session)
  43. define('STUDENT_BOSS', 17); // student is boss
  44. define('INVITEE', 20);
  45. define('HRM_REQUEST', 21); //HRM has request for vinculation with user
  46. // Table of status
  47. $_status_list[COURSEMANAGER] = 'teacher'; // 1
  48. $_status_list[SESSIONADMIN] = 'session_admin'; // 3
  49. $_status_list[DRH] = 'drh'; // 4
  50. $_status_list[STUDENT] = 'user'; // 5
  51. $_status_list[ANONYMOUS] = 'anonymous'; // 6
  52. $_status_list[INVITEE] = 'invited'; // 20
  53. // COURSE VISIBILITY CONSTANTS
  54. /** only visible for course admin */
  55. define('COURSE_VISIBILITY_CLOSED', 0);
  56. /** only visible for users registered in the course */
  57. define('COURSE_VISIBILITY_REGISTERED', 1);
  58. /** Open for all registered users on the platform */
  59. define('COURSE_VISIBILITY_OPEN_PLATFORM', 2);
  60. /** Open for the whole world */
  61. define('COURSE_VISIBILITY_OPEN_WORLD', 3);
  62. /** Invisible to all except admin */
  63. define('COURSE_VISIBILITY_HIDDEN', 4);
  64. define('COURSE_REQUEST_PENDING', 0);
  65. define('COURSE_REQUEST_ACCEPTED', 1);
  66. define('COURSE_REQUEST_REJECTED', 2);
  67. define('DELETE_ACTION_ENABLED', false);
  68. // EMAIL SENDING RECIPIENT CONSTANTS
  69. define('SEND_EMAIL_EVERYONE', 1);
  70. define('SEND_EMAIL_STUDENTS', 2);
  71. define('SEND_EMAIL_TEACHERS', 3);
  72. // SESSION VISIBILITY CONSTANTS
  73. define('SESSION_VISIBLE_READ_ONLY', 1);
  74. define('SESSION_VISIBLE', 2);
  75. define('SESSION_INVISIBLE', 3); // not available
  76. define('SESSION_AVAILABLE', 4);
  77. define('SESSION_LINK_TARGET', '_self');
  78. define('SUBSCRIBE_ALLOWED', 1);
  79. define('SUBSCRIBE_NOT_ALLOWED', 0);
  80. define('UNSUBSCRIBE_ALLOWED', 1);
  81. define('UNSUBSCRIBE_NOT_ALLOWED', 0);
  82. // SURVEY VISIBILITY CONSTANTS
  83. define('SURVEY_VISIBLE_TUTOR', 0);
  84. define('SURVEY_VISIBLE_TUTOR_STUDENT', 1);
  85. define('SURVEY_VISIBLE_PUBLIC', 2);
  86. // CONSTANTS defining all tools, using the english version
  87. /* When you add a new tool you must add it into function api_get_tools_lists() too */
  88. define('TOOL_DOCUMENT', 'document');
  89. define('TOOL_LP_FINAL_ITEM', 'final_item');
  90. define('TOOL_READOUT_TEXT', 'readout_text');
  91. define('TOOL_THUMBNAIL', 'thumbnail');
  92. define('TOOL_HOTPOTATOES', 'hotpotatoes');
  93. define('TOOL_CALENDAR_EVENT', 'calendar_event');
  94. define('TOOL_LINK', 'link');
  95. define('TOOL_LINK_CATEGORY', 'link_category');
  96. define('TOOL_COURSE_DESCRIPTION', 'course_description');
  97. define('TOOL_SEARCH', 'search');
  98. define('TOOL_LEARNPATH', 'learnpath');
  99. define('TOOL_LEARNPATH_CATEGORY', 'learnpath_category');
  100. define('TOOL_AGENDA', 'agenda');
  101. define('TOOL_ANNOUNCEMENT', 'announcement');
  102. define('TOOL_FORUM', 'forum');
  103. define('TOOL_FORUM_CATEGORY', 'forum_category');
  104. define('TOOL_FORUM_THREAD', 'forum_thread');
  105. define('TOOL_FORUM_POST', 'forum_post');
  106. define('TOOL_FORUM_ATTACH', 'forum_attachment');
  107. define('TOOL_FORUM_THREAD_QUALIFY', 'forum_thread_qualify');
  108. define('TOOL_THREAD', 'thread');
  109. define('TOOL_POST', 'post');
  110. define('TOOL_DROPBOX', 'dropbox');
  111. define('TOOL_QUIZ', 'quiz');
  112. define('TOOL_TEST_CATEGORY', 'test_category');
  113. define('TOOL_USER', 'user');
  114. define('TOOL_GROUP', 'group');
  115. define('TOOL_BLOGS', 'blog_management');
  116. define('TOOL_CHAT', 'chat');
  117. define('TOOL_STUDENTPUBLICATION', 'student_publication');
  118. define('TOOL_TRACKING', 'tracking');
  119. define('TOOL_HOMEPAGE_LINK', 'homepage_link');
  120. define('TOOL_COURSE_SETTING', 'course_setting');
  121. define('TOOL_BACKUP', 'backup');
  122. define('TOOL_COPY_COURSE_CONTENT', 'copy_course_content');
  123. define('TOOL_RECYCLE_COURSE', 'recycle_course');
  124. define('TOOL_COURSE_HOMEPAGE', 'course_homepage');
  125. define('TOOL_COURSE_RIGHTS_OVERVIEW', 'course_rights');
  126. define('TOOL_UPLOAD', 'file_upload');
  127. define('TOOL_COURSE_MAINTENANCE', 'course_maintenance');
  128. define('TOOL_SURVEY', 'survey');
  129. define('TOOL_WIKI', 'wiki');
  130. define('TOOL_GLOSSARY', 'glossary');
  131. define('TOOL_GRADEBOOK', 'gradebook');
  132. define('TOOL_NOTEBOOK', 'notebook');
  133. define('TOOL_ATTENDANCE', 'attendance');
  134. define('TOOL_COURSE_PROGRESS', 'course_progress');
  135. define('TOOL_PORTFOLIO', 'portfolio');
  136. define('TOOL_PLAGIARISM', 'compilatio');
  137. // CONSTANTS defining Chamilo interface sections
  138. define('SECTION_CAMPUS', 'mycampus');
  139. define('SECTION_COURSES', 'mycourses');
  140. define('SECTION_CATALOG', 'catalog');
  141. define('SECTION_MYPROFILE', 'myprofile');
  142. define('SECTION_MYAGENDA', 'myagenda');
  143. define('SECTION_COURSE_ADMIN', 'course_admin');
  144. define('SECTION_PLATFORM_ADMIN', 'platform_admin');
  145. define('SECTION_MYGRADEBOOK', 'mygradebook');
  146. define('SECTION_TRACKING', 'session_my_space');
  147. define('SECTION_SOCIAL', 'social-network');
  148. define('SECTION_DASHBOARD', 'dashboard');
  149. define('SECTION_REPORTS', 'reports');
  150. define('SECTION_GLOBAL', 'global');
  151. define('SECTION_INCLUDE', 'include');
  152. // CONSTANT name for local authentication source
  153. define('PLATFORM_AUTH_SOURCE', 'platform');
  154. define('CAS_AUTH_SOURCE', 'cas');
  155. define('LDAP_AUTH_SOURCE', 'extldap');
  156. // CONSTANT defining the default HotPotatoes files directory
  157. define('DIR_HOTPOTATOES', '/HotPotatoes_files');
  158. // event logs types
  159. define('LOG_COURSE_DELETE', 'course_deleted');
  160. define('LOG_COURSE_CREATE', 'course_created');
  161. // @todo replace 'soc_gr' with social_group
  162. define('LOG_GROUP_PORTAL_CREATED', 'soc_gr_created');
  163. define('LOG_GROUP_PORTAL_UPDATED', 'soc_gr_updated');
  164. define('LOG_GROUP_PORTAL_DELETED', 'soc_gr_deleted');
  165. define('LOG_GROUP_PORTAL_USER_DELETE_ALL', 'soc_gr_delete_users');
  166. define('LOG_GROUP_PORTAL_ID', 'soc_gr_portal_id');
  167. define('LOG_GROUP_PORTAL_REL_USER_ARRAY', 'soc_gr_user_array');
  168. define('LOG_GROUP_PORTAL_USER_SUBSCRIBED', 'soc_gr_u_subs');
  169. define('LOG_GROUP_PORTAL_USER_UNSUBSCRIBED', 'soc_gr_u_unsubs');
  170. define('LOG_GROUP_PORTAL_USER_UPDATE_ROLE', 'soc_gr_update_role');
  171. define('LOG_USER_DELETE', 'user_deleted');
  172. define('LOG_USER_CREATE', 'user_created');
  173. define('LOG_USER_ENABLE', 'user_enable');
  174. define('LOG_USER_DISABLE', 'user_disable');
  175. define('LOG_USER_ANONYMIZE', 'user_anonymized');
  176. define('LOG_USER_FIELD_CREATE', 'user_field_created');
  177. define('LOG_USER_FIELD_DELETE', 'user_field_deleted');
  178. define('LOG_SESSION_CREATE', 'session_created');
  179. define('LOG_SESSION_DELETE', 'session_deleted');
  180. define('LOG_SESSION_ADD_USER_COURSE', 'session_add_user_course');
  181. define('LOG_SESSION_DELETE_USER_COURSE', 'session_delete_user_course');
  182. define('LOG_SESSION_ADD_USER', 'session_add_user');
  183. define('LOG_SESSION_DELETE_USER', 'session_delete_user');
  184. define('LOG_SESSION_ADD_COURSE', 'session_add_course');
  185. define('LOG_SESSION_DELETE_COURSE', 'session_delete_course');
  186. define('LOG_SESSION_CATEGORY_CREATE', 'session_cat_created'); //changed in 1.9.8
  187. define('LOG_SESSION_CATEGORY_DELETE', 'session_cat_deleted'); //changed in 1.9.8
  188. define('LOG_CONFIGURATION_SETTINGS_CHANGE', 'settings_changed');
  189. define('LOG_PLATFORM_LANGUAGE_CHANGE', 'platform_lng_changed'); //changed in 1.9.8
  190. define('LOG_SUBSCRIBE_USER_TO_COURSE', 'user_subscribed');
  191. define('LOG_UNSUBSCRIBE_USER_FROM_COURSE', 'user_unsubscribed');
  192. define('LOG_ATTEMPTED_FORCED_LOGIN', 'attempted_forced_login');
  193. define('LOG_HOMEPAGE_CHANGED', 'homepage_changed');
  194. define('LOG_PROMOTION_CREATE', 'promotion_created');
  195. define('LOG_PROMOTION_DELETE', 'promotion_deleted');
  196. define('LOG_CAREER_CREATE', 'career_created');
  197. define('LOG_CAREER_DELETE', 'career_deleted');
  198. define('LOG_USER_PERSONAL_DOC_DELETED', 'user_doc_deleted');
  199. define('LOG_WIKI_ACCESS', 'wiki_page_view');
  200. // All results from an exercise
  201. define('LOG_EXERCISE_RESULT_DELETE', 'exe_result_deleted');
  202. // Logs only the one attempt
  203. define('LOG_EXERCISE_ATTEMPT_DELETE', 'exe_attempt_deleted');
  204. define('LOG_LP_ATTEMPT_DELETE', 'lp_attempt_deleted');
  205. define('LOG_QUESTION_RESULT_DELETE', 'qst_attempt_deleted');
  206. define('LOG_MY_FOLDER_CREATE', 'my_folder_created');
  207. define('LOG_MY_FOLDER_CHANGE', 'my_folder_changed');
  208. define('LOG_MY_FOLDER_DELETE', 'my_folder_deleted');
  209. define('LOG_MY_FOLDER_COPY', 'my_folder_copied');
  210. define('LOG_MY_FOLDER_CUT', 'my_folder_cut');
  211. define('LOG_MY_FOLDER_PASTE', 'my_folder_pasted');
  212. define('LOG_MY_FOLDER_UPLOAD', 'my_folder_uploaded');
  213. // Event logs data types (max 20 chars)
  214. define('LOG_COURSE_CODE', 'course_code');
  215. define('LOG_COURSE_ID', 'course_id');
  216. define('LOG_USER_ID', 'user_id');
  217. define('LOG_USER_OBJECT', 'user_object');
  218. define('LOG_USER_FIELD_VARIABLE', 'user_field_variable');
  219. define('LOG_SESSION_ID', 'session_id');
  220. define('LOG_QUESTION_ID', 'question_id');
  221. define('LOG_SESSION_CATEGORY_ID', 'session_category_id');
  222. define('LOG_CONFIGURATION_SETTINGS_CATEGORY', 'settings_category');
  223. define('LOG_CONFIGURATION_SETTINGS_VARIABLE', 'settings_variable');
  224. define('LOG_PLATFORM_LANGUAGE', 'default_platform_language');
  225. define('LOG_CAREER_ID', 'career_id');
  226. define('LOG_PROMOTION_ID', 'promotion_id');
  227. define('LOG_GRADEBOOK_LOCKED', 'gradebook_locked');
  228. define('LOG_GRADEBOOK_UNLOCKED', 'gradebook_unlocked');
  229. define('LOG_GRADEBOOK_ID', 'gradebook_id');
  230. define('LOG_WIKI_PAGE_ID', 'wiki_page_id');
  231. define('LOG_EXERCISE_ID', 'exercise_id');
  232. define('LOG_EXERCISE_AND_USER_ID', 'exercise_and_user_id');
  233. define('LOG_LP_ID', 'lp_id');
  234. define('LOG_EXERCISE_ATTEMPT_QUESTION_ID', 'exercise_a_q_id');
  235. define('LOG_EXERCISE_ATTEMPT', 'exe_id');
  236. define('LOG_WORK_DIR_DELETE', 'work_dir_delete');
  237. define('LOG_WORK_FILE_DELETE', 'work_file_delete');
  238. define('LOG_WORK_DATA', 'work_data_array');
  239. define('LOG_MY_FOLDER_PATH', 'path');
  240. define('LOG_MY_FOLDER_NEW_PATH', 'new_path');
  241. define('LOG_TERM_CONDITION_ACCEPTED', 'term_condition_accepted');
  242. define('LOG_USER_CONFIRMED_EMAIL', 'user_confirmed_email');
  243. define('LOG_USER_REMOVED_LEGAL_ACCEPT', 'user_removed_legal_accept');
  244. define('LOG_USER_DELETE_ACCOUNT_REQUEST', 'user_delete_account_request');
  245. define('LOG_QUESTION_CREATED', 'question_created');
  246. define('LOG_QUESTION_UPDATED', 'question_updated');
  247. define('USERNAME_PURIFIER', '/[^0-9A-Za-z_\.]/');
  248. //used when login_is_email setting is true
  249. define('USERNAME_PURIFIER_MAIL', '/[^0-9A-Za-z_\.@]/');
  250. define('USERNAME_PURIFIER_SHALLOW', '/\s/');
  251. // This constant is a result of Windows OS detection, it has a boolean value:
  252. // true whether the server runs on Windows OS, false otherwise.
  253. define('IS_WINDOWS_OS', api_is_windows_os());
  254. // Checks for installed optional php-extensions.
  255. // intl extension (from PECL), it is installed by default as of PHP 5.3.0.
  256. define('INTL_INSTALLED', function_exists('intl_get_error_code'));
  257. // iconv extension, for PHP5 on Windows it is installed by default.
  258. define('ICONV_INSTALLED', function_exists('iconv'));
  259. define('MBSTRING_INSTALLED', function_exists('mb_strlen')); // mbstring extension.
  260. // Patterns for processing paths. Examples.
  261. define('REPEATED_SLASHES_PURIFIER', '/\/{2,}/'); // $path = preg_replace(REPEATED_SLASHES_PURIFIER, '/', $path);
  262. define('VALID_WEB_PATH', '/https?:\/\/[^\/]*(\/.*)?/i'); // $is_valid_path = preg_match(VALID_WEB_PATH, $path);
  263. // $new_path = preg_replace(VALID_WEB_SERVER_BASE, $new_base, $path);
  264. define('VALID_WEB_SERVER_BASE', '/https?:\/\/[^\/]*/i');
  265. // Constants for api_get_path() and api_get_path_type(), etc. - registered path types.
  266. // basic (leaf elements)
  267. define('REL_CODE_PATH', 'REL_CODE_PATH');
  268. define('REL_COURSE_PATH', 'REL_COURSE_PATH');
  269. define('REL_HOME_PATH', 'REL_HOME_PATH');
  270. // Constants for api_get_path() and api_get_path_type(), etc. - registered path types.
  271. define('WEB_PATH', 'WEB_PATH');
  272. define('WEB_APP_PATH', 'WEB_APP_PATH');
  273. define('SYS_PATH', 'SYS_PATH');
  274. define('SYS_APP_PATH', 'SYS_APP_PATH');
  275. define('SYS_UPLOAD_PATH', 'SYS_UPLOAD_PATH');
  276. define('WEB_UPLOAD_PATH', 'WEB_UPLOAD_PATH');
  277. define('REL_PATH', 'REL_PATH');
  278. define('WEB_COURSE_PATH', 'WEB_COURSE_PATH');
  279. define('SYS_COURSE_PATH', 'SYS_COURSE_PATH');
  280. define('WEB_CODE_PATH', 'WEB_CODE_PATH');
  281. define('SYS_CODE_PATH', 'SYS_CODE_PATH');
  282. define('SYS_LANG_PATH', 'SYS_LANG_PATH');
  283. define('WEB_IMG_PATH', 'WEB_IMG_PATH');
  284. define('WEB_CSS_PATH', 'WEB_CSS_PATH');
  285. define('WEB_PUBLIC_PATH', 'WEB_PUBLIC_PATH');
  286. define('SYS_CSS_PATH', 'SYS_CSS_PATH');
  287. define('SYS_PLUGIN_PATH', 'SYS_PLUGIN_PATH');
  288. define('WEB_PLUGIN_PATH', 'WEB_PLUGIN_PATH');
  289. define('WEB_PLUGIN_ASSET_PATH', 'WEB_PLUGIN_ASSET_PATH');
  290. define('SYS_ARCHIVE_PATH', 'SYS_ARCHIVE_PATH');
  291. define('WEB_ARCHIVE_PATH', 'WEB_ARCHIVE_PATH');
  292. define('SYS_INC_PATH', 'SYS_INC_PATH');
  293. define('LIBRARY_PATH', 'LIBRARY_PATH');
  294. define('CONFIGURATION_PATH', 'CONFIGURATION_PATH');
  295. define('WEB_LIBRARY_PATH', 'WEB_LIBRARY_PATH');
  296. define('WEB_LIBRARY_JS_PATH', 'WEB_LIBRARY_JS_PATH');
  297. define('WEB_AJAX_PATH', 'WEB_AJAX_PATH');
  298. define('SYS_TEST_PATH', 'SYS_TEST_PATH');
  299. define('WEB_TEMPLATE_PATH', 'WEB_TEMPLATE_PATH');
  300. define('SYS_TEMPLATE_PATH', 'SYS_TEMPLATE_PATH');
  301. define('SYS_PUBLIC_PATH', 'SYS_PUBLIC_PATH');
  302. define('SYS_HOME_PATH', 'SYS_HOME_PATH');
  303. define('WEB_HOME_PATH', 'WEB_HOME_PATH');
  304. define('WEB_FONTS_PATH', 'WEB_FONTS_PATH');
  305. define('SYS_FONTS_PATH', 'SYS_FONTS_PATH');
  306. define('SYS_DEFAULT_COURSE_DOCUMENT_PATH', 'SYS_DEFAULT_COURSE_DOCUMENT_PATH');
  307. define('REL_DEFAULT_COURSE_DOCUMENT_PATH', 'REL_DEFAULT_COURSE_DOCUMENT_PATH');
  308. define('WEB_DEFAULT_COURSE_DOCUMENT_PATH', 'WEB_DEFAULT_COURSE_DOCUMENT_PATH');
  309. // Relations type with Course manager
  310. define('COURSE_RELATION_TYPE_COURSE_MANAGER', 1);
  311. define('SESSION_RELATION_TYPE_COURSE_MANAGER', 1);
  312. // Relations type with Human resources manager
  313. define('COURSE_RELATION_TYPE_RRHH', 1);
  314. define('SESSION_RELATION_TYPE_RRHH', 1);
  315. //User image sizes
  316. define('USER_IMAGE_SIZE_ORIGINAL', 1);
  317. define('USER_IMAGE_SIZE_BIG', 2);
  318. define('USER_IMAGE_SIZE_MEDIUM', 3);
  319. define('USER_IMAGE_SIZE_SMALL', 4);
  320. // Relation type between users
  321. define('USER_UNKNOWN', 0);
  322. define('USER_RELATION_TYPE_UNKNOWN', 1);
  323. define('USER_RELATION_TYPE_PARENT', 2); // should be deprecated is useless
  324. define('USER_RELATION_TYPE_FRIEND', 3);
  325. define('USER_RELATION_TYPE_GOODFRIEND', 4); // should be deprecated is useless
  326. define('USER_RELATION_TYPE_ENEMY', 5); // should be deprecated is useless
  327. define('USER_RELATION_TYPE_DELETED', 6);
  328. define('USER_RELATION_TYPE_RRHH', 7);
  329. define('USER_RELATION_TYPE_BOSS', 8);
  330. define('USER_RELATION_TYPE_HRM_REQUEST', 9);
  331. // Gradebook link constants
  332. // Please do not change existing values, they are used in the database !
  333. define('GRADEBOOK_ITEM_LIMIT', 1000);
  334. define('LINK_EXERCISE', 1);
  335. define('LINK_DROPBOX', 2);
  336. define('LINK_STUDENTPUBLICATION', 3);
  337. define('LINK_LEARNPATH', 4);
  338. define('LINK_FORUM_THREAD', 5);
  339. //define('LINK_WORK',6);
  340. define('LINK_ATTENDANCE', 7);
  341. define('LINK_SURVEY', 8);
  342. define('LINK_HOTPOTATOES', 9);
  343. // Score display types constants
  344. define('SCORE_DIV', 1); // X / Y
  345. define('SCORE_PERCENT', 2); // XX %
  346. define('SCORE_DIV_PERCENT', 3); // X / Y (XX %)
  347. define('SCORE_AVERAGE', 4); // XX %
  348. define('SCORE_DECIMAL', 5); // 0.50 (X/Y)
  349. define('SCORE_BAR', 6); // Uses the Display::bar_progress function
  350. define('SCORE_SIMPLE', 7); // X
  351. define('SCORE_IGNORE_SPLIT', 8); // ??
  352. define('SCORE_DIV_PERCENT_WITH_CUSTOM', 9); // X / Y (XX %) - Good!
  353. define('SCORE_CUSTOM', 10); // Good!
  354. define('SCORE_DIV_SIMPLE_WITH_CUSTOM', 11); // X - Good!
  355. define('SCORE_DIV_SIMPLE_WITH_CUSTOM_LETTERS', 12); // X - Good!
  356. define('SCORE_ONLY_SCORE', 13); // X - Good!
  357. define('SCORE_NUMERIC', 14);
  358. define('SCORE_BOTH', 1);
  359. define('SCORE_ONLY_DEFAULT', 2);
  360. define('SCORE_ONLY_CUSTOM', 3);
  361. // From display.lib.php
  362. define('MAX_LENGTH_BREADCRUMB', 100);
  363. define('ICON_SIZE_ATOM', 8);
  364. define('ICON_SIZE_TINY', 16);
  365. define('ICON_SIZE_SMALL', 22);
  366. define('ICON_SIZE_MEDIUM', 32);
  367. define('ICON_SIZE_LARGE', 48);
  368. define('ICON_SIZE_BIG', 64);
  369. define('ICON_SIZE_HUGE', 128);
  370. define('SHOW_TEXT_NEAR_ICONS', false);
  371. // Session catalog
  372. define('CATALOG_COURSES', 0);
  373. define('CATALOG_SESSIONS', 1);
  374. define('CATALOG_COURSES_SESSIONS', 2);
  375. // Hook type events, pre-process and post-process.
  376. // All means to be executed for both hook event types
  377. define('HOOK_EVENT_TYPE_PRE', 0);
  378. define('HOOK_EVENT_TYPE_POST', 1);
  379. define('HOOK_EVENT_TYPE_ALL', 10);
  380. define('CAREER_STATUS_ACTIVE', 1);
  381. define('CAREER_STATUS_INACTIVE', 0);
  382. define('PROMOTION_STATUS_ACTIVE', 1);
  383. define('PROMOTION_STATUS_INACTIVE', 0);
  384. // Group permissions
  385. define('GROUP_PERMISSION_OPEN', '1');
  386. define('GROUP_PERMISSION_CLOSED', '2');
  387. // Group user permissions
  388. define('GROUP_USER_PERMISSION_ADMIN', '1'); // the admin of a group
  389. define('GROUP_USER_PERMISSION_READER', '2'); // a normal user
  390. define('GROUP_USER_PERMISSION_PENDING_INVITATION', '3'); // When an admin/moderator invites a user
  391. define('GROUP_USER_PERMISSION_PENDING_INVITATION_SENT_BY_USER', '4'); // an user joins a group
  392. define('GROUP_USER_PERMISSION_MODERATOR', '5'); // a moderator
  393. define('GROUP_USER_PERMISSION_ANONYMOUS', '6'); // an anonymous user
  394. define('GROUP_USER_PERMISSION_HRM', '7'); // a human resources manager
  395. define('GROUP_IMAGE_SIZE_ORIGINAL', 1);
  396. define('GROUP_IMAGE_SIZE_BIG', 2);
  397. define('GROUP_IMAGE_SIZE_MEDIUM', 3);
  398. define('GROUP_IMAGE_SIZE_SMALL', 4);
  399. define('GROUP_TITLE_LENGTH', 50);
  400. // Exercise
  401. // @todo move into a class
  402. define('ALL_ON_ONE_PAGE', 1);
  403. define('ONE_PER_PAGE', 2);
  404. define('EXERCISE_FEEDBACK_TYPE_END', 0); //Feedback - show score and expected answers
  405. define('EXERCISE_FEEDBACK_TYPE_DIRECT', 1); //DirectFeedback - Do not show score nor answers
  406. define('EXERCISE_FEEDBACK_TYPE_EXAM', 2); // NoFeedback - Show score only
  407. define('EXERCISE_FEEDBACK_TYPE_POPUP', 3); // Popup BT#15827
  408. define('RESULT_DISABLE_SHOW_SCORE_AND_EXPECTED_ANSWERS', 0); //show score and expected answers
  409. define('RESULT_DISABLE_NO_SCORE_AND_EXPECTED_ANSWERS', 1); //Do not show score nor answers
  410. define('RESULT_DISABLE_SHOW_SCORE_ONLY', 2); //Show score only
  411. define('RESULT_DISABLE_SHOW_FINAL_SCORE_ONLY_WITH_CATEGORIES', 3); //Show final score only with categories
  412. define('RESULT_DISABLE_SHOW_SCORE_ATTEMPT_SHOW_ANSWERS_LAST_ATTEMPT', 4);
  413. define('RESULT_DISABLE_DONT_SHOW_SCORE_ONLY_IF_USER_FINISHES_ATTEMPTS_SHOW_ALWAYS_FEEDBACK', 5);
  414. define('RESULT_DISABLE_RANKING', 6);
  415. define('RESULT_DISABLE_SHOW_ONLY_IN_CORRECT_ANSWER', 7);
  416. define('RESULT_DISABLE_SHOW_SCORE_AND_EXPECTED_ANSWERS_AND_RANKING', 8);
  417. define('EXERCISE_MAX_NAME_SIZE', 80);
  418. // Question types (edit next array as well when adding values)
  419. // @todo move into a class
  420. define('UNIQUE_ANSWER', 1);
  421. define('MULTIPLE_ANSWER', 2);
  422. define('FILL_IN_BLANKS', 3);
  423. define('MATCHING', 4);
  424. define('FREE_ANSWER', 5);
  425. define('HOT_SPOT', 6);
  426. define('HOT_SPOT_ORDER', 7);
  427. define('HOT_SPOT_DELINEATION', 8);
  428. define('MULTIPLE_ANSWER_COMBINATION', 9);
  429. define('UNIQUE_ANSWER_NO_OPTION', 10);
  430. define('MULTIPLE_ANSWER_TRUE_FALSE', 11);
  431. define('MULTIPLE_ANSWER_COMBINATION_TRUE_FALSE', 12);
  432. define('ORAL_EXPRESSION', 13);
  433. define('GLOBAL_MULTIPLE_ANSWER', 14);
  434. define('MEDIA_QUESTION', 15);
  435. define('CALCULATED_ANSWER', 16);
  436. define('UNIQUE_ANSWER_IMAGE', 17);
  437. define('DRAGGABLE', 18);
  438. define('MATCHING_DRAGGABLE', 19);
  439. define('ANNOTATION', 20);
  440. define('READING_COMPREHENSION', 21);
  441. define('MULTIPLE_ANSWER_TRUE_FALSE_DEGREE_CERTAINTY', 22);
  442. define('EXERCISE_CATEGORY_RANDOM_SHUFFLED', 1);
  443. define('EXERCISE_CATEGORY_RANDOM_ORDERED', 2);
  444. define('EXERCISE_CATEGORY_RANDOM_DISABLED', 0);
  445. // Question selection type
  446. define('EX_Q_SELECTION_ORDERED', 1);
  447. define('EX_Q_SELECTION_RANDOM', 2);
  448. define('EX_Q_SELECTION_CATEGORIES_ORDERED_QUESTIONS_ORDERED', 3);
  449. define('EX_Q_SELECTION_CATEGORIES_RANDOM_QUESTIONS_ORDERED', 4);
  450. define('EX_Q_SELECTION_CATEGORIES_ORDERED_QUESTIONS_RANDOM', 5);
  451. define('EX_Q_SELECTION_CATEGORIES_RANDOM_QUESTIONS_RANDOM', 6);
  452. define('EX_Q_SELECTION_CATEGORIES_RANDOM_QUESTIONS_ORDERED_NO_GROUPED', 7);
  453. define('EX_Q_SELECTION_CATEGORIES_RANDOM_QUESTIONS_RANDOM_NO_GROUPED', 8);
  454. define('EX_Q_SELECTION_CATEGORIES_ORDERED_BY_PARENT_QUESTIONS_ORDERED', 9);
  455. define('EX_Q_SELECTION_CATEGORIES_ORDERED_BY_PARENT_QUESTIONS_RANDOM', 10);
  456. // Used to save the skill_rel_item table
  457. define('ITEM_TYPE_EXERCISE', 1);
  458. define('ITEM_TYPE_HOTPOTATOES', 2);
  459. define('ITEM_TYPE_LINK', 3);
  460. define('ITEM_TYPE_LEARNPATH', 4);
  461. define('ITEM_TYPE_GRADEBOOK', 5);
  462. define('ITEM_TYPE_STUDENT_PUBLICATION', 6);
  463. //define('ITEM_TYPE_FORUM', 7);
  464. define('ITEM_TYPE_ATTENDANCE', 8);
  465. define('ITEM_TYPE_SURVEY', 9);
  466. define('ITEM_TYPE_FORUM_THREAD', 10);
  467. // one big string with all question types, for the validator in pear/HTML/QuickForm/Rule/QuestionType
  468. define(
  469. 'QUESTION_TYPES',
  470. UNIQUE_ANSWER.':'.
  471. MULTIPLE_ANSWER.':'.
  472. FILL_IN_BLANKS.':'.
  473. MATCHING.':'.
  474. FREE_ANSWER.':'.
  475. HOT_SPOT.':'.
  476. HOT_SPOT_ORDER.':'.
  477. HOT_SPOT_DELINEATION.':'.
  478. MULTIPLE_ANSWER_COMBINATION.':'.
  479. UNIQUE_ANSWER_NO_OPTION.':'.
  480. MULTIPLE_ANSWER_TRUE_FALSE.':'.
  481. MULTIPLE_ANSWER_COMBINATION_TRUE_FALSE.':'.
  482. ORAL_EXPRESSION.':'.
  483. GLOBAL_MULTIPLE_ANSWER.':'.
  484. MEDIA_QUESTION.':'.
  485. CALCULATED_ANSWER.':'.
  486. UNIQUE_ANSWER_IMAGE.':'.
  487. DRAGGABLE.':'.
  488. MATCHING_DRAGGABLE.':'.
  489. MULTIPLE_ANSWER_TRUE_FALSE_DEGREE_CERTAINTY.':'.
  490. ANNOTATION
  491. );
  492. //Some alias used in the QTI exports
  493. define('MCUA', 1);
  494. define('TF', 1);
  495. define('MCMA', 2);
  496. define('FIB', 3);
  497. // Skills
  498. define('SKILL_TYPE_REQUIREMENT', 'required');
  499. define('SKILL_TYPE_ACQUIRED', 'acquired');
  500. define('SKILL_TYPE_BOTH', 'both');
  501. // Message
  502. define('MESSAGE_STATUS_NEW', '0');
  503. define('MESSAGE_STATUS_UNREAD', '1');
  504. //2 ??
  505. define('MESSAGE_STATUS_DELETED', '3');
  506. define('MESSAGE_STATUS_OUTBOX', '4');
  507. define('MESSAGE_STATUS_INVITATION_PENDING', '5');
  508. define('MESSAGE_STATUS_INVITATION_ACCEPTED', '6');
  509. define('MESSAGE_STATUS_INVITATION_DENIED', '7');
  510. define('MESSAGE_STATUS_WALL', '8');
  511. define('MESSAGE_STATUS_WALL_DELETE', '9');
  512. define('MESSAGE_STATUS_WALL_POST', '10');
  513. define('MESSAGE_STATUS_CONVERSATION', '11');
  514. define('MESSAGE_STATUS_FORUM', '12');
  515. define('MESSAGE_STATUS_PROMOTED', '13');
  516. // Images
  517. define('IMAGE_WALL_SMALL_SIZE', 200);
  518. define('IMAGE_WALL_MEDIUM_SIZE', 500);
  519. define('IMAGE_WALL_BIG_SIZE', 2000);
  520. define('IMAGE_WALL_SMALL', 'small');
  521. define('IMAGE_WALL_MEDIUM', 'medium');
  522. define('IMAGE_WALL_BIG', 'big');
  523. // Social PLUGIN PLACES
  524. define('SOCIAL_LEFT_PLUGIN', 1);
  525. define('SOCIAL_CENTER_PLUGIN', 2);
  526. define('SOCIAL_RIGHT_PLUGIN', 3);
  527. define('CUT_GROUP_NAME', 50);
  528. /**
  529. * FormValidator Filter.
  530. */
  531. define('NO_HTML', 1);
  532. define('STUDENT_HTML', 2);
  533. define('TEACHER_HTML', 3);
  534. define('STUDENT_HTML_FULLPAGE', 4);
  535. define('TEACHER_HTML_FULLPAGE', 5);
  536. // Timeline
  537. define('TIMELINE_STATUS_ACTIVE', '1');
  538. define('TIMELINE_STATUS_INACTIVE', '2');
  539. // Event email template class
  540. define('EVENT_EMAIL_TEMPLATE_ACTIVE', 1);
  541. define('EVENT_EMAIL_TEMPLATE_INACTIVE', 0);
  542. // Course home
  543. define('SHORTCUTS_HORIZONTAL', 0);
  544. define('SHORTCUTS_VERTICAL', 1);
  545. // Image class
  546. define('IMAGE_PROCESSOR', 'gd'); // 'imagick' or 'gd' strings
  547. // Course copy
  548. define('FILE_SKIP', 1);
  549. define('FILE_RENAME', 2);
  550. define('FILE_OVERWRITE', 3);
  551. define('UTF8_CONVERT', false); //false by default
  552. define('DOCUMENT', 'file');
  553. define('FOLDER', 'folder');
  554. define('RESOURCE_ASSET', 'asset');
  555. define('RESOURCE_DOCUMENT', 'document');
  556. define('RESOURCE_GLOSSARY', 'glossary');
  557. define('RESOURCE_EVENT', 'calendar_event');
  558. define('RESOURCE_LINK', 'link');
  559. define('RESOURCE_COURSEDESCRIPTION', 'course_description');
  560. define('RESOURCE_LEARNPATH', 'learnpath');
  561. define('RESOURCE_LEARNPATH_CATEGORY', 'learnpath_category');
  562. define('RESOURCE_ANNOUNCEMENT', 'announcement');
  563. define('RESOURCE_FORUM', 'forum');
  564. define('RESOURCE_FORUMTOPIC', 'thread');
  565. define('RESOURCE_FORUMPOST', 'post');
  566. define('RESOURCE_QUIZ', 'quiz');
  567. define('RESOURCE_TEST_CATEGORY', 'test_category');
  568. define('RESOURCE_QUIZQUESTION', 'Exercise_Question');
  569. define('RESOURCE_TOOL_INTRO', 'Tool introduction');
  570. define('RESOURCE_LINKCATEGORY', 'Link_Category');
  571. define('RESOURCE_FORUMCATEGORY', 'Forum_Category');
  572. define('RESOURCE_SCORM', 'Scorm');
  573. define('RESOURCE_SURVEY', 'survey');
  574. define('RESOURCE_SURVEYQUESTION', 'survey_question');
  575. define('RESOURCE_SURVEYINVITATION', 'survey_invitation');
  576. define('RESOURCE_WIKI', 'wiki');
  577. define('RESOURCE_THEMATIC', 'thematic');
  578. define('RESOURCE_ATTENDANCE', 'attendance');
  579. define('RESOURCE_WORK', 'work');
  580. define('RESOURCE_SESSION_COURSE', 'session_course');
  581. define('RESOURCE_GRADEBOOK', 'gradebook');
  582. define('ADD_THEMATIC_PLAN', 6);
  583. // Max online users to show per page (whoisonline)
  584. define('MAX_ONLINE_USERS', 12);
  585. // Number of characters maximum to show in preview of course blog posts
  586. define('BLOG_MAX_PREVIEW_CHARS', 800);
  587. // HTML string to replace with a 'Read more...' link
  588. define('BLOG_PAGE_BREAK', '<div style="page-break-after: always"><span style="display: none;">&nbsp;</span></div>');
  589. // Make sure the CHAMILO_LOAD_WYSIWYG constant is defined
  590. // To remove CKeditor libs from HTML, set this constant to true before loading
  591. if (!defined('CHAMILO_LOAD_WYSIWYG')) {
  592. define('CHAMILO_LOAD_WYSIWYG', true);
  593. }
  594. /* Constants for course home */
  595. define('TOOL_PUBLIC', 'Public');
  596. define('TOOL_PUBLIC_BUT_HIDDEN', 'PublicButHide');
  597. define('TOOL_COURSE_ADMIN', 'courseAdmin');
  598. define('TOOL_PLATFORM_ADMIN', 'platformAdmin');
  599. define('TOOL_AUTHORING', 'toolauthoring');
  600. define('TOOL_INTERACTION', 'toolinteraction');
  601. define('TOOL_COURSE_PLUGIN', 'toolcourseplugin'); //all plugins that can be enabled in courses
  602. define('TOOL_ADMIN', 'tooladmin');
  603. define('TOOL_ADMIN_PLATFORM', 'tooladminplatform');
  604. define('TOOL_DRH', 'tool_drh');
  605. define('TOOL_STUDENT_VIEW', 'toolstudentview');
  606. define('TOOL_ADMIN_VISIBLE', 'tooladminvisible');
  607. /**
  608. * Inclusion of internationalization libraries.
  609. */
  610. require_once __DIR__.'/internationalization.lib.php';
  611. /**
  612. * Returns a path to a certain resource within the Chamilo area, specifyed through a parameter.
  613. * Also, this function provides conversion between path types, in this case the input path points inside the Chamilo area too.
  614. *
  615. * See $_configuration['course_folder'] in the configuration.php to alter the WEB_COURSE_PATH and SYS_COURSE_PATH parameters.
  616. *
  617. * @param string $path (optional) A path which type is to be converted. Also, it may be a defined constant for a path.
  618. * This parameter has meaning when $type parameter has one of the following values: TO_WEB, TO_SYS, TO_REL. Otherwise it is ignored.
  619. *
  620. * @return string the requested path or the converted path
  621. *
  622. *
  623. * Notes about the current behaviour model:
  624. * 1. Windows back-slashes are converted to slashes in the result.
  625. * 2. A semi-absolute web-path is detected by its leading slash. On Linux systems, absolute system paths start with
  626. * a slash too, so an additional check about presence of leading system server base is implemented. For example, the function is
  627. * able to distinguish type difference between /var/www/chamilo/courses/ (SYS) and /chamilo/courses/ (REL).
  628. * 3. The function api_get_path() returns only these three types of paths, which in some sense are absolute. The function has
  629. * no a mechanism for processing relative web/system paths, such as: lesson01.html, ./lesson01.html, ../css/my_styles.css.
  630. * It has not been identified as needed yet.
  631. * 4. Also, resolving the meta-symbols "." and ".." within paths has not been implemented, it is to be identified as needed.
  632. *
  633. * For examples go to: *
  634. * See main/admin/system_status.php?section=paths
  635. *
  636. * Vchamilo changes : allow using an alternate configuration
  637. * to get vchamilo instance paths
  638. */
  639. function api_get_path($path = '', $configuration = [])
  640. {
  641. global $paths;
  642. // get proper configuration data if exists
  643. global $_configuration;
  644. $emptyConfigurationParam = false;
  645. if (empty($configuration)) {
  646. $configuration = (array) $_configuration;
  647. $emptyConfigurationParam = true;
  648. }
  649. $course_folder = 'courses/';
  650. static $root_web = '';
  651. $root_sys = $_configuration['root_sys'];
  652. // If no $root_web has been set so far *and* no custom config has been passed to the function
  653. // then re-use the previously-calculated (run-specific) $root_web and skip this complex calculation
  654. if (empty($root_web) || $emptyConfigurationParam === false || empty($configuration)) {
  655. // Resolve master hostname.
  656. if (!empty($configuration) && array_key_exists('root_web', $configuration)) {
  657. $root_web = $configuration['root_web'];
  658. } else {
  659. $root_web = '';
  660. // Try guess it from server.
  661. if (defined('SYSTEM_INSTALLATION') && SYSTEM_INSTALLATION) {
  662. if (($pos = strpos(($requested_page_rel = api_get_self()), 'main/install')) !== false) {
  663. $root_rel = substr($requested_page_rel, 0, $pos);
  664. // See http://www.mediawiki.org/wiki/Manual:$wgServer
  665. $server_protocol = (isset($_SERVER['HTTPS']) && $_SERVER['HTTPS'] == 'on') ? 'https' : 'http';
  666. $server_name =
  667. isset($_SERVER['SERVER_NAME']) ? $_SERVER['SERVER_NAME']
  668. : (isset($_SERVER['HOSTNAME']) ? $_SERVER['HOSTNAME']
  669. : (isset($_SERVER['HTTP_HOST']) ? $_SERVER['HTTP_HOST']
  670. : (isset($_SERVER['SERVER_ADDR']) ? $_SERVER['SERVER_ADDR']
  671. : 'localhost')));
  672. if (isset($_SERVER['SERVER_PORT']) &&
  673. !strpos($server_name, ':') &&
  674. (($server_protocol == 'http' && $_SERVER['SERVER_PORT'] != 80) ||
  675. ($server_protocol == 'https' && $_SERVER['SERVER_PORT'] != 443))
  676. ) {
  677. $server_name .= ":".$_SERVER['SERVER_PORT'];
  678. }
  679. $root_web = $server_protocol.'://'.$server_name.$root_rel;
  680. $root_sys = str_replace('\\', '/', realpath(__DIR__.'/../../../')).'/';
  681. }
  682. // Here we give up, so we don't touch anything.
  683. }
  684. }
  685. }
  686. if (isset($configuration['multiple_access_urls']) &&
  687. $configuration['multiple_access_urls']
  688. ) {
  689. // To avoid that the api_get_access_url() function fails since global.inc.php also calls the api.lib.php.
  690. if (isset($configuration['access_url']) && !empty($configuration['access_url'])) {
  691. // We look into the DB the function api_get_access_url
  692. $urlInfo = api_get_access_url($configuration['access_url']);
  693. // Avoid default value
  694. $defaultValues = ['http://localhost/', 'https://localhost/'];
  695. if (!empty($urlInfo['url']) && !in_array($urlInfo['url'], $defaultValues)) {
  696. $root_web = $urlInfo['active'] == 1 ? $urlInfo['url'] : $configuration['root_web'];
  697. }
  698. }
  699. }
  700. $paths = [];
  701. // Initialise cache with default values.
  702. if (!array_key_exists($root_web, $paths)) {
  703. $paths[$root_web] = [
  704. WEB_PATH => '',
  705. SYS_PATH => '',
  706. REL_PATH => '',
  707. WEB_COURSE_PATH => '',
  708. SYS_COURSE_PATH => '',
  709. REL_COURSE_PATH => '',
  710. WEB_CODE_PATH => 'main/',
  711. SYS_CODE_PATH => 'main/',
  712. REL_CODE_PATH => '/main/',
  713. SYS_LANG_PATH => 'lang/',
  714. WEB_IMG_PATH => 'img/',
  715. WEB_CSS_PATH => 'web/css/',
  716. SYS_CSS_PATH => 'app/Resources/public/css/',
  717. SYS_PLUGIN_PATH => 'plugin/',
  718. WEB_PLUGIN_PATH => 'plugin/',
  719. WEB_PLUGIN_ASSET_PATH => 'public/plugins/',
  720. SYS_ARCHIVE_PATH => 'app/cache/',
  721. WEB_ARCHIVE_PATH => 'app/cache/',
  722. SYS_HOME_PATH => 'app/home/',
  723. WEB_HOME_PATH => 'app/home/',
  724. REL_HOME_PATH => 'app/home/',
  725. SYS_APP_PATH => 'app/',
  726. WEB_APP_PATH => 'app/',
  727. SYS_UPLOAD_PATH => 'app/upload/',
  728. SYS_INC_PATH => 'inc/',
  729. CONFIGURATION_PATH => 'app/config/',
  730. LIBRARY_PATH => 'inc/lib/',
  731. WEB_LIBRARY_PATH => 'inc/lib/',
  732. WEB_LIBRARY_JS_PATH => 'inc/lib/javascript/',
  733. WEB_AJAX_PATH => 'inc/ajax/',
  734. SYS_TEST_PATH => 'tests/',
  735. WEB_TEMPLATE_PATH => 'template/',
  736. SYS_TEMPLATE_PATH => 'template/',
  737. WEB_UPLOAD_PATH => 'app/upload/',
  738. WEB_PUBLIC_PATH => 'web/',
  739. SYS_PUBLIC_PATH => 'web/',
  740. WEB_FONTS_PATH => 'fonts/',
  741. SYS_FONTS_PATH => 'fonts/',
  742. ];
  743. }
  744. $isInitialized = [];
  745. $course_folder = isset($configuration['course_folder']) ? $configuration['course_folder'] : $course_folder;
  746. $root_rel = isset($configuration['url_append']) ? $configuration['url_append'] : '';
  747. if (!empty($root_rel)) {
  748. // Adds "/" to the root_rel
  749. $hasSlash = substr($configuration['url_append'], 0, 1);
  750. if ($hasSlash !== '/') {
  751. $root_rel = '/'.$root_rel;
  752. }
  753. }
  754. // Web server base and system server base.
  755. if (!array_key_exists($root_web, $isInitialized)) {
  756. // process absolute global roots
  757. if (!empty($configuration)) {
  758. $code_folder = 'main';
  759. } else {
  760. $code_folder = $paths[$root_web][REL_CODE_PATH];
  761. }
  762. // Support for the installation process.
  763. // Developers might use the function api_get_path() directly or indirectly (this is difficult to be traced), at the moment when
  764. // configuration has not been created yet. This is why this function should be upgraded to return correct results in this case.
  765. // Dealing with trailing slashes.
  766. $slashed_root_web = api_add_trailing_slash($root_web);
  767. $root_sys = api_add_trailing_slash($root_sys);
  768. $root_rel = api_add_trailing_slash($root_rel);
  769. $code_folder = api_add_trailing_slash($code_folder);
  770. $course_folder = api_add_trailing_slash($course_folder);
  771. // Initialization of a table that contains common-purpose paths.
  772. $paths[$root_web][REL_PATH] = $root_rel;
  773. $paths[$root_web][REL_COURSE_PATH] = $root_rel.$course_folder;
  774. $paths[$root_web][REL_CODE_PATH] = $root_rel.$code_folder;
  775. $paths[$root_web][REL_DEFAULT_COURSE_DOCUMENT_PATH] = $paths[$root_web][REL_PATH].'main/default_course_document/';
  776. $paths[$root_web][WEB_PATH] = $slashed_root_web;
  777. $paths[$root_web][WEB_CODE_PATH] = $paths[$root_web][WEB_PATH].$code_folder;
  778. $paths[$root_web][WEB_COURSE_PATH] = $paths[$root_web][WEB_PATH].$course_folder;
  779. $paths[$root_web][WEB_DEFAULT_COURSE_DOCUMENT_PATH] = $paths[$root_web][WEB_CODE_PATH].'default_course_document/';
  780. $paths[$root_web][WEB_APP_PATH] = $paths[$root_web][WEB_PATH].$paths[$root_web][WEB_APP_PATH];
  781. $paths[$root_web][WEB_PLUGIN_PATH] = $paths[$root_web][WEB_PATH].$paths[$root_web][WEB_PLUGIN_PATH];
  782. $paths[$root_web][WEB_PLUGIN_ASSET_PATH] = $paths[$root_web][WEB_PATH].$paths[$root_web][WEB_PLUGIN_ASSET_PATH];
  783. $paths[$root_web][WEB_ARCHIVE_PATH] = $paths[$root_web][WEB_PATH].$paths[$root_web][WEB_ARCHIVE_PATH];
  784. $paths[$root_web][WEB_CSS_PATH] = $paths[$root_web][WEB_PATH].$paths[$root_web][WEB_CSS_PATH];
  785. $paths[$root_web][WEB_IMG_PATH] = $paths[$root_web][WEB_CODE_PATH].$paths[$root_web][WEB_IMG_PATH];
  786. $paths[$root_web][WEB_LIBRARY_PATH] = $paths[$root_web][WEB_CODE_PATH].$paths[$root_web][WEB_LIBRARY_PATH];
  787. $paths[$root_web][WEB_LIBRARY_JS_PATH] = $paths[$root_web][WEB_CODE_PATH].$paths[$root_web][WEB_LIBRARY_JS_PATH];
  788. $paths[$root_web][WEB_AJAX_PATH] = $paths[$root_web][WEB_CODE_PATH].$paths[$root_web][WEB_AJAX_PATH];
  789. $paths[$root_web][WEB_FONTS_PATH] = $paths[$root_web][WEB_CODE_PATH].$paths[$root_web][WEB_FONTS_PATH];
  790. $paths[$root_web][WEB_TEMPLATE_PATH] = $paths[$root_web][WEB_CODE_PATH].$paths[$root_web][WEB_TEMPLATE_PATH];
  791. $paths[$root_web][WEB_UPLOAD_PATH] = $paths[$root_web][WEB_PATH].$paths[$root_web][WEB_UPLOAD_PATH];
  792. $paths[$root_web][WEB_PUBLIC_PATH] = $paths[$root_web][WEB_PATH].$paths[$root_web][WEB_PUBLIC_PATH];
  793. $paths[$root_web][WEB_HOME_PATH] = $paths[$root_web][WEB_PATH].$paths[$root_web][REL_HOME_PATH];
  794. $paths[$root_web][SYS_PATH] = $root_sys;
  795. $paths[$root_web][SYS_CODE_PATH] = $root_sys.$code_folder;
  796. $paths[$root_web][SYS_TEST_PATH] = $paths[$root_web][SYS_PATH].$paths[$root_web][SYS_TEST_PATH];
  797. $paths[$root_web][SYS_TEMPLATE_PATH] = $paths[$root_web][SYS_CODE_PATH].$paths[$root_web][SYS_TEMPLATE_PATH];
  798. $paths[$root_web][SYS_PUBLIC_PATH] = $paths[$root_web][SYS_PATH].$paths[$root_web][SYS_PUBLIC_PATH];
  799. $paths[$root_web][SYS_CSS_PATH] = $paths[$root_web][SYS_PATH].$paths[$root_web][SYS_CSS_PATH];
  800. $paths[$root_web][SYS_FONTS_PATH] = $paths[$root_web][SYS_CODE_PATH].$paths[$root_web][SYS_FONTS_PATH];
  801. $paths[$root_web][SYS_ARCHIVE_PATH] = $paths[$root_web][SYS_PATH].$paths[$root_web][SYS_ARCHIVE_PATH];
  802. $paths[$root_web][SYS_APP_PATH] = $paths[$root_web][SYS_PATH].$paths[$root_web][SYS_APP_PATH];
  803. $paths[$root_web][SYS_COURSE_PATH] = $paths[$root_web][SYS_APP_PATH].$course_folder;
  804. $paths[$root_web][SYS_UPLOAD_PATH] = $paths[$root_web][SYS_PATH].$paths[$root_web][SYS_UPLOAD_PATH];
  805. $paths[$root_web][SYS_LANG_PATH] = $paths[$root_web][SYS_CODE_PATH].$paths[$root_web][SYS_LANG_PATH];
  806. $paths[$root_web][SYS_HOME_PATH] = $paths[$root_web][SYS_PATH].$paths[$root_web][SYS_HOME_PATH];
  807. $paths[$root_web][SYS_PLUGIN_PATH] = $paths[$root_web][SYS_PATH].$paths[$root_web][SYS_PLUGIN_PATH];
  808. $paths[$root_web][SYS_INC_PATH] = $paths[$root_web][SYS_CODE_PATH].$paths[$root_web][SYS_INC_PATH];
  809. $paths[$root_web][LIBRARY_PATH] = $paths[$root_web][SYS_CODE_PATH].$paths[$root_web][LIBRARY_PATH];
  810. $paths[$root_web][CONFIGURATION_PATH] = $paths[$root_web][SYS_PATH].$paths[$root_web][CONFIGURATION_PATH];
  811. global $virtualChamilo;
  812. if (!empty($virtualChamilo)) {
  813. $paths[$root_web][SYS_ARCHIVE_PATH] = api_add_trailing_slash($virtualChamilo[SYS_ARCHIVE_PATH]);
  814. $paths[$root_web][SYS_HOME_PATH] = api_add_trailing_slash($virtualChamilo[SYS_HOME_PATH]);
  815. $paths[$root_web][SYS_COURSE_PATH] = api_add_trailing_slash($virtualChamilo[SYS_COURSE_PATH]);
  816. $paths[$root_web][SYS_UPLOAD_PATH] = api_add_trailing_slash($virtualChamilo[SYS_UPLOAD_PATH]);
  817. $paths[$root_web][WEB_HOME_PATH] = api_add_trailing_slash($virtualChamilo[WEB_HOME_PATH]);
  818. $paths[$root_web][WEB_UPLOAD_PATH] = api_add_trailing_slash($virtualChamilo[WEB_UPLOAD_PATH]);
  819. $paths[$root_web][WEB_ARCHIVE_PATH] = api_add_trailing_slash($virtualChamilo[WEB_ARCHIVE_PATH]);
  820. //$paths[$root_web][WEB_COURSE_PATH] = api_add_trailing_slash($virtualChamilo[WEB_COURSE_PATH]);
  821. // WEB_UPLOAD_PATH should be handle by apache htaccess in the vhost
  822. // RewriteEngine On
  823. // RewriteRule /app/upload/(.*)$ http://localhost/other/upload/my-chamilo111-net/$1 [QSA,L]
  824. //$paths[$root_web][WEB_UPLOAD_PATH] = api_add_trailing_slash($virtualChamilo[WEB_UPLOAD_PATH]);
  825. //$paths[$root_web][REL_PATH] = $virtualChamilo[REL_PATH];
  826. //$paths[$root_web][REL_COURSE_PATH] = $virtualChamilo[REL_COURSE_PATH];
  827. }
  828. $isInitialized[$root_web] = true;
  829. }
  830. $path = trim($path);
  831. // Retrieving a common-purpose path.
  832. if (isset($paths[$root_web][$path])) {
  833. return $paths[$root_web][$path];
  834. }
  835. // Second purification.
  836. // Replacing Windows back slashes.
  837. $path = str_replace('\\', '/', $path);
  838. // Query strings sometimes mighth wrongly appear in non-URLs.
  839. // Let us check remove them from all types of paths.
  840. if (($pos = strpos($path, '?')) !== false) {
  841. $path = substr($path, 0, $pos);
  842. }
  843. // Detection of the input path type. Conversion to semi-absolute type ( /chamilo/main/inc/.... ).
  844. if (preg_match(VALID_WEB_PATH, $path)) {
  845. // A special case: When a URL points to the document download script directly, without
  846. // mod-rewrite translation, we have to translate it into an "ordinary" web path.
  847. // For example:
  848. // http://localhost/chamilo/main/document/download.php?doc_url=/image.png&cDir=/
  849. // becomes
  850. // http://localhost/chamilo/courses/TEST/document/image.png
  851. // TEST is a course directory name, so called "system course code".
  852. if (strpos($path, 'download.php') !== false) { // Fast detection first.
  853. $path = urldecode($path);
  854. if (preg_match('/(.*)main\/document\/download.php\?doc_url=\/(.*)&cDir=\/(.*)?/', $path, $matches)) {
  855. $sys_course_code =
  856. isset($_SESSION['_course']['sysCode']) // User is inside a course?
  857. ? $_SESSION['_course']['sysCode'] // Yes, then use course's directory name.
  858. : '{SYS_COURSE_CODE}'; // No, then use a fake code, it may be processed later.
  859. $path = $matches[1].'courses/'.$sys_course_code.'/document/'.str_replace('//', '/', $matches[3].'/'.$matches[2]);
  860. }
  861. }
  862. // Replacement of the present web server base with a slash '/'.
  863. $path = preg_replace(VALID_WEB_SERVER_BASE, '/', $path);
  864. }
  865. // Path now is semi-absolute. It is convenient at this moment repeated slashes to be removed.
  866. $path = preg_replace(REPEATED_SLASHES_PURIFIER, '/', $path);
  867. return $path;
  868. }
  869. /**
  870. * Gets a modified version of the path for the CDN, if defined in
  871. * configuration.php.
  872. *
  873. * @param string $web_path The path of the resource without CDN
  874. *
  875. * @return string The path of the resource converted to CDN
  876. *
  877. * @author Yannick Warnier <ywarnier@beeznst.org>
  878. */
  879. function api_get_cdn_path($web_path)
  880. {
  881. global $_configuration;
  882. if (!empty($_configuration['cdn_enable'])) {
  883. $web_root = api_get_path(WEB_PATH);
  884. $ext = substr($web_path, strrpos($web_path, '.'));
  885. if (isset($ext[2])) { // faster version of strlen to check if len>2
  886. // Check for CDN definitions
  887. if (!empty($ext)) {
  888. foreach ($_configuration['cdn'] as $host => $exts) {
  889. if (in_array($ext, $exts)) {
  890. //Use host as defined in $_configuration['cdn'], without
  891. // trailing slash
  892. return str_replace($web_root, $host.'/', $web_path);
  893. }
  894. }
  895. }
  896. }
  897. }
  898. return $web_path;
  899. }
  900. /**
  901. * @return bool Return true if CAS authentification is activated
  902. */
  903. function api_is_cas_activated()
  904. {
  905. return api_get_setting('cas_activate') == "true";
  906. }
  907. /**
  908. * @return bool Return true if LDAP authentification is activated
  909. */
  910. function api_is_ldap_activated()
  911. {
  912. global $extAuthSource;
  913. return is_array($extAuthSource[LDAP_AUTH_SOURCE]);
  914. }
  915. /**
  916. * @return bool Return true if Facebook authentification is activated
  917. */
  918. function api_is_facebook_auth_activated()
  919. {
  920. global $_configuration;
  921. return isset($_configuration['facebook_auth']) && $_configuration['facebook_auth'] == 1;
  922. }
  923. /**
  924. * Adds to a given path a trailing slash if it is necessary (adds "/" character at the end of the string).
  925. *
  926. * @param string $path the input path
  927. *
  928. * @return string returns the modified path
  929. */
  930. function api_add_trailing_slash($path)
  931. {
  932. return substr($path, -1) == '/' ? $path : $path.'/';
  933. }
  934. /**
  935. * Removes from a given path the trailing slash if it is necessary (removes "/" character from the end of the string).
  936. *
  937. * @param string $path the input path
  938. *
  939. * @return string returns the modified path
  940. */
  941. function api_remove_trailing_slash($path)
  942. {
  943. return substr($path, -1) == '/' ? substr($path, 0, -1) : $path;
  944. }
  945. /**
  946. * Checks the RFC 3986 syntax of a given URL.
  947. *
  948. * @param string $url the URL to be checked
  949. * @param bool $absolute whether the URL is absolute (beginning with a scheme such as "http:")
  950. *
  951. * @return string|false Returns the URL if it is valid, FALSE otherwise.
  952. * This function is an adaptation from the function valid_url(), Drupal CMS.
  953. *
  954. * @see http://drupal.org
  955. * Note: The built-in function filter_var($urs, FILTER_VALIDATE_URL) has a bug for some versions of PHP.
  956. * @see http://bugs.php.net/51192
  957. */
  958. function api_valid_url($url, $absolute = false)
  959. {
  960. if ($absolute) {
  961. if (preg_match("
  962. /^ # Start at the beginning of the text
  963. (?:ftp|https?|feed):\/\/ # Look for ftp, http, https or feed schemes
  964. (?: # Userinfo (optional) which is typically
  965. (?:(?:[\w\.\-\+!$&'\(\)*\+,;=]|%[0-9a-f]{2})+:)* # a username or a username and password
  966. (?:[\w\.\-\+%!$&'\(\)*\+,;=]|%[0-9a-f]{2})+@ # combination
  967. )?
  968. (?:
  969. (?:[a-z0-9\-\.]|%[0-9a-f]{2})+ # A domain name or a IPv4 address
  970. |(?:\[(?:[0-9a-f]{0,4}:)*(?:[0-9a-f]{0,4})\]) # or a well formed IPv6 address
  971. )
  972. (?::[0-9]+)? # Server port number (optional)
  973. (?:[\/|\?]
  974. (?:[\w#!:\.\?\+=&@$'~*,;\/\(\)\[\]\-]|%[0-9a-f]{2}) # The path and query (optional)
  975. *)?
  976. $/xi", $url)) {
  977. return $url;
  978. }
  979. return false;
  980. } else {
  981. return preg_match("/^(?:[\w#!:\.\?\+=&@$'~*,;\/\(\)\[\]\-]|%[0-9a-f]{2})+$/i", $url) ? $url : false;
  982. }
  983. }
  984. /**
  985. * Checks whether a given string looks roughly like an email address.
  986. *
  987. * @param string $address the e-mail address to be checked
  988. *
  989. * @return mixed returns the e-mail if it is valid, FALSE otherwise
  990. */
  991. function api_valid_email($address)
  992. {
  993. return filter_var($address, FILTER_VALIDATE_EMAIL);
  994. }
  995. /* PROTECTION FUNCTIONS
  996. Use these functions to protect your scripts. */
  997. /**
  998. * Function used to protect a course script.
  999. * The function blocks access when
  1000. * - there is no $_SESSION["_course"] defined; or
  1001. * - $is_allowed_in_course is set to false (this depends on the course
  1002. * visibility and user status).
  1003. *
  1004. * This is only the first proposal, test and improve!
  1005. *
  1006. * @param bool Option to print headers when displaying error message. Default: false
  1007. * @param bool whether session admins should be allowed or not
  1008. * @param bool $checkTool check if tool is available for users (user, group)
  1009. *
  1010. * @return bool True if the user has access to the current course or is out of a course context, false otherwise
  1011. *
  1012. * @todo replace global variable
  1013. *
  1014. * @author Roan Embrechts
  1015. */
  1016. function api_protect_course_script($print_headers = false, $allow_session_admins = false, $checkTool = '')
  1017. {
  1018. $course_info = api_get_course_info();
  1019. if (empty($course_info)) {
  1020. api_not_allowed($print_headers);
  1021. return false;
  1022. }
  1023. if (api_is_drh()) {
  1024. return true;
  1025. }
  1026. // Session admin has access to course
  1027. $sessionAccess = api_get_configuration_value('session_admins_access_all_content');
  1028. if ($sessionAccess) {
  1029. $allow_session_admins = true;
  1030. }
  1031. if (api_is_platform_admin($allow_session_admins)) {
  1032. return true;
  1033. }
  1034. $isAllowedInCourse = api_is_allowed_in_course();
  1035. $is_visible = false;
  1036. if (isset($course_info) && isset($course_info['visibility'])) {
  1037. switch ($course_info['visibility']) {
  1038. default:
  1039. case COURSE_VISIBILITY_CLOSED:
  1040. // Completely closed: the course is only accessible to the teachers. - 0
  1041. if (api_get_user_id() && !api_is_anonymous() && $isAllowedInCourse) {
  1042. $is_visible = true;
  1043. }
  1044. break;
  1045. case COURSE_VISIBILITY_REGISTERED:
  1046. // Private - access authorized to course members only - 1
  1047. if (api_get_user_id() && !api_is_anonymous() && $isAllowedInCourse) {
  1048. $is_visible = true;
  1049. }
  1050. break;
  1051. case COURSE_VISIBILITY_OPEN_PLATFORM:
  1052. // Open - access allowed for users registered on the platform - 2
  1053. if (api_get_user_id() && !api_is_anonymous() && $isAllowedInCourse) {
  1054. $is_visible = true;
  1055. }
  1056. break;
  1057. case COURSE_VISIBILITY_OPEN_WORLD:
  1058. //Open - access allowed for the whole world - 3
  1059. $is_visible = true;
  1060. break;
  1061. case COURSE_VISIBILITY_HIDDEN:
  1062. //Completely closed: the course is only accessible to the teachers. - 0
  1063. if (api_is_platform_admin()) {
  1064. $is_visible = true;
  1065. }
  1066. break;
  1067. }
  1068. //If password is set and user is not registered to the course then the course is not visible
  1069. if ($isAllowedInCourse == false &&
  1070. isset($course_info['registration_code']) &&
  1071. !empty($course_info['registration_code'])
  1072. ) {
  1073. $is_visible = false;
  1074. }
  1075. }
  1076. if (!empty($checkTool)) {
  1077. if (!api_is_allowed_to_edit(true, true, true)) {
  1078. $toolInfo = api_get_tool_information_by_name($checkTool);
  1079. if (!empty($toolInfo) && isset($toolInfo['visibility']) && $toolInfo['visibility'] == 0) {
  1080. api_not_allowed(true);
  1081. return false;
  1082. }
  1083. }
  1084. }
  1085. // Check session visibility
  1086. $session_id = api_get_session_id();
  1087. if (!empty($session_id)) {
  1088. // $isAllowedInCourse was set in local.inc.php
  1089. if (!$isAllowedInCourse) {
  1090. $is_visible = false;
  1091. }
  1092. }
  1093. if (!$is_visible) {
  1094. api_not_allowed($print_headers);
  1095. return false;
  1096. }
  1097. return true;
  1098. }
  1099. /**
  1100. * Function used to protect an admin script.
  1101. *
  1102. * The function blocks access when the user has no platform admin rights
  1103. * with an error message printed on default output
  1104. *
  1105. * @param bool Whether to allow session admins as well
  1106. * @param bool Whether to allow HR directors as well
  1107. * @param string An optional message (already passed through get_lang)
  1108. *
  1109. * @return bool True if user is allowed, false otherwise.
  1110. * The function also outputs an error message in case not allowed
  1111. *
  1112. * @author Roan Embrechts (original author)
  1113. */
  1114. function api_protect_admin_script($allow_sessions_admins = false, $allow_drh = false, $message = null)
  1115. {
  1116. if (!api_is_platform_admin($allow_sessions_admins, $allow_drh)) {
  1117. api_not_allowed(true, $message);
  1118. return false;
  1119. }
  1120. return true;
  1121. }
  1122. /**
  1123. * Function used to protect a teacher script.
  1124. * The function blocks access when the user has no teacher rights.
  1125. *
  1126. * @return bool True if the current user can access the script, false otherwise
  1127. *
  1128. * @author Yoselyn Castillo
  1129. */
  1130. function api_protect_teacher_script()
  1131. {
  1132. if (!api_is_allowed_to_edit()) {
  1133. api_not_allowed(true);
  1134. return false;
  1135. }
  1136. return true;
  1137. }
  1138. /**
  1139. * Function used to prevent anonymous users from accessing a script.
  1140. *
  1141. * @param bool|true $printHeaders
  1142. *
  1143. * @author Roan Embrechts
  1144. *
  1145. * @return bool
  1146. */
  1147. function api_block_anonymous_users($printHeaders = true)
  1148. {
  1149. $user = api_get_user_info();
  1150. if (!(isset($user['user_id']) && $user['user_id']) || api_is_anonymous($user['user_id'], true)) {
  1151. api_not_allowed($printHeaders);
  1152. return false;
  1153. }
  1154. return true;
  1155. }
  1156. /**
  1157. * Returns a rough evaluation of the browser's name and version based on very
  1158. * simple regexp.
  1159. *
  1160. * @return array with the navigator name and version ['name' => '...', 'version' => '...']
  1161. */
  1162. function api_get_navigator()
  1163. {
  1164. $navigator = 'Unknown';
  1165. $version = 0;
  1166. if (!isset($_SERVER['HTTP_USER_AGENT'])) {
  1167. return ['name' => 'Unknown', 'version' => '0.0.0'];
  1168. }
  1169. if (strpos($_SERVER['HTTP_USER_AGENT'], 'Opera') !== false) {
  1170. $navigator = 'Opera';
  1171. list(, $version) = explode('Opera', $_SERVER['HTTP_USER_AGENT']);
  1172. } elseif (strpos($_SERVER['HTTP_USER_AGENT'], 'Edge') !== false) {
  1173. $navigator = 'Edge';
  1174. list(, $version) = explode('Edge', $_SERVER['HTTP_USER_AGENT']);
  1175. } elseif (strpos($_SERVER['HTTP_USER_AGENT'], 'MSIE') !== false) {
  1176. $navigator = 'Internet Explorer';
  1177. list(, $version) = explode('MSIE ', $_SERVER['HTTP_USER_AGENT']);
  1178. } elseif (strpos($_SERVER['HTTP_USER_AGENT'], 'Chrome') !== false) {
  1179. $navigator = 'Chrome';
  1180. list(, $version) = explode('Chrome', $_SERVER['HTTP_USER_AGENT']);
  1181. } elseif (stripos($_SERVER['HTTP_USER_AGENT'], 'Safari') !== false) {
  1182. $navigator = 'Safari';
  1183. if (stripos($_SERVER['HTTP_USER_AGENT'], 'Version/') !== false) {
  1184. // If this Safari does have the "Version/" string in its user agent
  1185. // then use that as a version indicator rather than what's after
  1186. // "Safari/" which is rather a "build number" or something
  1187. list(, $version) = explode('Version/', $_SERVER['HTTP_USER_AGENT']);
  1188. } else {
  1189. list(, $version) = explode('Safari/', $_SERVER['HTTP_USER_AGENT']);
  1190. }
  1191. } elseif (strpos($_SERVER['HTTP_USER_AGENT'], 'Firefox') !== false) {
  1192. $navigator = 'Firefox';
  1193. list(, $version) = explode('Firefox', $_SERVER['HTTP_USER_AGENT']);
  1194. } elseif (strpos($_SERVER['HTTP_USER_AGENT'], 'Netscape') !== false) {
  1195. $navigator = 'Netscape';
  1196. if (stripos($_SERVER['HTTP_USER_AGENT'], 'Netscape/') !== false) {
  1197. list(, $version) = explode('Netscape', $_SERVER['HTTP_USER_AGENT']);
  1198. } else {
  1199. list(, $version) = explode('Navigator', $_SERVER['HTTP_USER_AGENT']);
  1200. }
  1201. } elseif (strpos($_SERVER['HTTP_USER_AGENT'], 'Konqueror') !== false) {
  1202. $navigator = 'Konqueror';
  1203. list(, $version) = explode('Konqueror', $_SERVER['HTTP_USER_AGENT']);
  1204. } elseif (stripos($_SERVER['HTTP_USER_AGENT'], 'applewebkit') !== false) {
  1205. $navigator = 'AppleWebKit';
  1206. list(, $version) = explode('Version/', $_SERVER['HTTP_USER_AGENT']);
  1207. } elseif (strpos($_SERVER['HTTP_USER_AGENT'], 'Gecko') !== false) {
  1208. $navigator = 'Mozilla';
  1209. list(, $version) = explode('; rv:', $_SERVER['HTTP_USER_AGENT']);
  1210. }
  1211. // Now cut extra stuff around (mostly *after*) the version number
  1212. $version = preg_replace('/^([\/\s])?([\d\.]+)?.*/', '\2', $version);
  1213. if (strpos($version, '.') === false) {
  1214. $version = number_format(doubleval($version), 1);
  1215. }
  1216. $return = ['name' => $navigator, 'version' => $version];
  1217. return $return;
  1218. }
  1219. /**
  1220. * @return true if user self registration is allowed, false otherwise
  1221. */
  1222. function api_is_self_registration_allowed()
  1223. {
  1224. return isset($GLOBALS['allowSelfReg']) ? $GLOBALS['allowSelfReg'] : false;
  1225. }
  1226. /**
  1227. * This function returns the id of the user which is stored in the $_user array.
  1228. *
  1229. * example: The function can be used to check if a user is logged in
  1230. * if (api_get_user_id())
  1231. *
  1232. * @return int the id of the current user, 0 if is empty
  1233. */
  1234. function api_get_user_id()
  1235. {
  1236. $userInfo = Session::read('_user');
  1237. if ($userInfo && isset($userInfo['user_id'])) {
  1238. return (int) $userInfo['user_id'];
  1239. }
  1240. return 0;
  1241. }
  1242. /**
  1243. * Gets the list of courses a specific user is subscribed to.
  1244. *
  1245. * @param int User ID
  1246. * @param bool $fetch_session Whether to get session courses or not - NOT YET IMPLEMENTED
  1247. *
  1248. * @return array Array of courses in the form [0]=>('code'=>xxx,'db'=>xxx,'dir'=>xxx,'status'=>d)
  1249. *
  1250. * @deprecated use CourseManager::get_courses_list_by_user_id()
  1251. */
  1252. function api_get_user_courses($userId, $fetch_session = true)
  1253. {
  1254. // Get out if not integer
  1255. if ($userId != strval(intval($userId))) {
  1256. return [];
  1257. }
  1258. $t_course = Database::get_main_table(TABLE_MAIN_COURSE);
  1259. $t_course_user = Database::get_main_table(TABLE_MAIN_COURSE_USER);
  1260. $sql = "SELECT cc.id as real_id, cc.code code, cc.directory dir, cu.status status
  1261. FROM $t_course cc, $t_course_user cu
  1262. WHERE
  1263. cc.id = cu.c_id AND
  1264. cu.user_id = $userId AND
  1265. cu.relation_type <> ".COURSE_RELATION_TYPE_RRHH;
  1266. $result = Database::query($sql);
  1267. if ($result === false) {
  1268. return [];
  1269. }
  1270. $courses = [];
  1271. while ($row = Database::fetch_array($result)) {
  1272. // we only need the database name of the course
  1273. $courses[] = $row;
  1274. }
  1275. return $courses;
  1276. }
  1277. /**
  1278. * Formats user information into a standard array
  1279. * This function should be only used inside api_get_user_info().
  1280. *
  1281. * @param array Non-standard user array
  1282. * @param bool $add_password
  1283. * @param bool $loadAvatars turn off to improve performance
  1284. *
  1285. * @return array Standard user array
  1286. */
  1287. function _api_format_user($user, $add_password = false, $loadAvatars = true)
  1288. {
  1289. $result = [];
  1290. $result['firstname'] = null;
  1291. $result['lastname'] = null;
  1292. if (isset($user['firstname']) && isset($user['lastname'])) { // with only lowercase
  1293. $result['firstname'] = $user['firstname'];
  1294. $result['lastname'] = $user['lastname'];
  1295. } elseif (isset($user['firstName']) && isset($user['lastName'])) { // with uppercase letters
  1296. $result['firstname'] = isset($user['firstName']) ? $user['firstName'] : null;
  1297. $result['lastname'] = isset($user['lastName']) ? $user['lastName'] : null;
  1298. }
  1299. if (isset($user['email'])) {
  1300. $result['mail'] = isset($user['email']) ? $user['email'] : null;
  1301. $result['email'] = isset($user['email']) ? $user['email'] : null;
  1302. } else {
  1303. $result['mail'] = isset($user['mail']) ? $user['mail'] : null;
  1304. $result['email'] = isset($user['mail']) ? $user['mail'] : null;
  1305. }
  1306. $result['complete_name'] = api_get_person_name($result['firstname'], $result['lastname']);
  1307. $result['complete_name_with_username'] = $result['complete_name'];
  1308. if (!empty($user['username']) && !api_get_configuration_value('hide_username_with_complete_name')) {
  1309. $result['complete_name_with_username'] = $result['complete_name'].' ('.$user['username'].')';
  1310. }
  1311. $showEmail = api_get_setting('show_email_addresses') === 'true';
  1312. if (!empty($user['email'])) {
  1313. $result['complete_name_with_email_forced'] = $result['complete_name'].' ('.$user['email'].')';
  1314. if ($showEmail) {
  1315. $result['complete_name_with_email'] = $result['complete_name'].' ('.$user['email'].')';
  1316. }
  1317. } else {
  1318. $result['complete_name_with_email'] = $result['complete_name'];
  1319. $result['complete_name_with_email_forced'] = $result['complete_name'];
  1320. }
  1321. // Kept for historical reasons
  1322. $result['firstName'] = $result['firstname'];
  1323. $result['lastName'] = $result['lastname'];
  1324. $attributes = [
  1325. 'phone',
  1326. 'address',
  1327. 'picture_uri',
  1328. 'official_code',
  1329. 'status',
  1330. 'active',
  1331. 'auth_source',
  1332. 'username',
  1333. 'theme',
  1334. 'language',
  1335. 'creator_id',
  1336. 'registration_date',
  1337. 'hr_dept_id',
  1338. 'expiration_date',
  1339. 'last_login',
  1340. 'user_is_online',
  1341. 'profile_completed',
  1342. ];
  1343. if (api_get_setting('extended_profile') === 'true') {
  1344. $attributes[] = 'competences';
  1345. $attributes[] = 'diplomas';
  1346. $attributes[] = 'teach';
  1347. $attributes[] = 'openarea';
  1348. }
  1349. foreach ($attributes as $attribute) {
  1350. $result[$attribute] = isset($user[$attribute]) ? $user[$attribute] : null;
  1351. }
  1352. $user_id = (int) $user['user_id'];
  1353. // Maintain the user_id index for backwards compatibility
  1354. $result['user_id'] = $result['id'] = $user_id;
  1355. $hasCertificates = Certificate::getCertificateByUser($user_id);
  1356. $result['has_certificates'] = 0;
  1357. if (!empty($hasCertificates)) {
  1358. $result['has_certificates'] = 1;
  1359. }
  1360. $result['icon_status'] = '';
  1361. $result['icon_status_medium'] = '';
  1362. $result['is_admin'] = UserManager::is_admin($user_id);
  1363. // Getting user avatar.
  1364. if ($loadAvatars) {
  1365. $result['avatar'] = '';
  1366. $result['avatar_no_query'] = '';
  1367. $result['avatar_small'] = '';
  1368. $result['avatar_medium'] = '';
  1369. if (!isset($user['avatar'])) {
  1370. $originalFile = UserManager::getUserPicture(
  1371. $user_id,
  1372. USER_IMAGE_SIZE_ORIGINAL,
  1373. null,
  1374. $result
  1375. );
  1376. $result['avatar'] = $originalFile;
  1377. $avatarString = explode('?', $result['avatar']);
  1378. $result['avatar_no_query'] = reset($avatarString);
  1379. } else {
  1380. $result['avatar'] = $user['avatar'];
  1381. $avatarString = explode('?', $user['avatar']);
  1382. $result['avatar_no_query'] = reset($avatarString);
  1383. }
  1384. if (!isset($user['avatar_small'])) {
  1385. $smallFile = UserManager::getUserPicture(
  1386. $user_id,
  1387. USER_IMAGE_SIZE_SMALL,
  1388. null,
  1389. $result
  1390. );
  1391. $result['avatar_small'] = $smallFile;
  1392. } else {
  1393. $result['avatar_small'] = $user['avatar_small'];
  1394. }
  1395. if (!isset($user['avatar_medium'])) {
  1396. $mediumFile = UserManager::getUserPicture(
  1397. $user_id,
  1398. USER_IMAGE_SIZE_MEDIUM,
  1399. null,
  1400. $result
  1401. );
  1402. $result['avatar_medium'] = $mediumFile;
  1403. } else {
  1404. $result['avatar_medium'] = $user['avatar_medium'];
  1405. }
  1406. $urlImg = api_get_path(WEB_IMG_PATH);
  1407. $iconStatus = '';
  1408. $iconStatusMedium = '';
  1409. switch ($result['status']) {
  1410. case STUDENT:
  1411. if ($result['has_certificates']) {
  1412. $iconStatus = $urlImg.'icons/svg/identifier_graduated.svg';
  1413. } else {
  1414. $iconStatus = $urlImg.'icons/svg/identifier_student.svg';
  1415. }
  1416. break;
  1417. case COURSEMANAGER:
  1418. if ($result['is_admin']) {
  1419. $iconStatus = $urlImg.'icons/svg/identifier_admin.svg';
  1420. } else {
  1421. $iconStatus = $urlImg.'icons/svg/identifier_teacher.svg';
  1422. }
  1423. break;
  1424. case STUDENT_BOSS:
  1425. $iconStatus = $urlImg.'icons/svg/identifier_teacher.svg';
  1426. break;
  1427. }
  1428. if (!empty($iconStatus)) {
  1429. $iconStatusMedium = '<img src="'.$iconStatus.'" width="32px" height="32px">';
  1430. $iconStatus = '<img src="'.$iconStatus.'" width="22px" height="22px">';
  1431. }
  1432. $result['icon_status'] = $iconStatus;
  1433. $result['icon_status_medium'] = $iconStatusMedium;
  1434. }
  1435. if (isset($user['user_is_online'])) {
  1436. $result['user_is_online'] = $user['user_is_online'] == true ? 1 : 0;
  1437. }
  1438. if (isset($user['user_is_online_in_chat'])) {
  1439. $result['user_is_online_in_chat'] = (int) $user['user_is_online_in_chat'];
  1440. }
  1441. if ($add_password) {
  1442. $result['password'] = $user['password'];
  1443. }
  1444. if (isset($result['profile_completed'])) {
  1445. $result['profile_completed'] = $user['profile_completed'];
  1446. }
  1447. $result['profile_url'] = api_get_path(WEB_CODE_PATH).'social/profile.php?u='.$user_id;
  1448. // Send message link
  1449. $sendMessage = api_get_path(WEB_AJAX_PATH).'user_manager.ajax.php?a=get_user_popup&user_id='.$user_id;
  1450. $result['complete_name_with_message_link'] = Display::url(
  1451. $result['complete_name_with_username'],
  1452. $sendMessage,
  1453. ['class' => 'ajax']
  1454. );
  1455. if (isset($user['extra'])) {
  1456. $result['extra'] = $user['extra'];
  1457. }
  1458. return $result;
  1459. }
  1460. /**
  1461. * Finds all the information about a user.
  1462. * If no parameter is passed you find all the information about the current user.
  1463. *
  1464. * @param int $user_id
  1465. * @param bool $checkIfUserOnline
  1466. * @param bool $showPassword
  1467. * @param bool $loadExtraData
  1468. * @param bool $loadOnlyVisibleExtraData Get the user extra fields that are visible
  1469. * @param bool $loadAvatars turn off to improve performance and if avatars are not needed
  1470. * @param bool $updateCache update apc cache if exists
  1471. *
  1472. * @return mixed $user_info user_id, lastname, firstname, username, email, etc or false on error
  1473. *
  1474. * @author Patrick Cool <patrick.cool@UGent.be>
  1475. * @author Julio Montoya
  1476. *
  1477. * @version 21 September 2004
  1478. */
  1479. function api_get_user_info(
  1480. $user_id = 0,
  1481. $checkIfUserOnline = false,
  1482. $showPassword = false,
  1483. $loadExtraData = false,
  1484. $loadOnlyVisibleExtraData = false,
  1485. $loadAvatars = true,
  1486. $updateCache = false
  1487. ) {
  1488. $apcVar = null;
  1489. $user = false;
  1490. $cacheAvailable = api_get_configuration_value('apc');
  1491. if (empty($user_id)) {
  1492. $userFromSession = Session::read('_user');
  1493. if (isset($userFromSession)) {
  1494. if ($cacheAvailable === true &&
  1495. (
  1496. empty($userFromSession['is_anonymous']) &&
  1497. (isset($userFromSession['status']) && $userFromSession['status'] != ANONYMOUS)
  1498. )
  1499. ) {
  1500. $apcVar = api_get_configuration_value('apc_prefix').'userinfo_'.$userFromSession['user_id'];
  1501. if (apcu_exists($apcVar)) {
  1502. if ($updateCache) {
  1503. apcu_store($apcVar, $userFromSession, 60);
  1504. }
  1505. $user = apcu_fetch($apcVar);
  1506. } else {
  1507. $user = _api_format_user(
  1508. $userFromSession,
  1509. $showPassword,
  1510. $loadAvatars
  1511. );
  1512. apcu_store($apcVar, $user, 60);
  1513. }
  1514. } else {
  1515. $user = _api_format_user(
  1516. $userFromSession,
  1517. $showPassword,
  1518. $loadAvatars
  1519. );
  1520. }
  1521. return $user;
  1522. }
  1523. return false;
  1524. }
  1525. // Make sure user_id is safe
  1526. $user_id = (int) $user_id;
  1527. // Re-use user information if not stale and already stored in APCu
  1528. if ($cacheAvailable === true) {
  1529. $apcVar = api_get_configuration_value('apc_prefix').'userinfo_'.$user_id;
  1530. if (apcu_exists($apcVar) && $updateCache == false && $checkIfUserOnline == false) {
  1531. $user = apcu_fetch($apcVar);
  1532. return $user;
  1533. }
  1534. }
  1535. $sql = "SELECT * FROM ".Database::get_main_table(TABLE_MAIN_USER)."
  1536. WHERE id = $user_id";
  1537. $result = Database::query($sql);
  1538. if (Database::num_rows($result) > 0) {
  1539. $result_array = Database::fetch_array($result);
  1540. $result_array['user_is_online_in_chat'] = 0;
  1541. if ($checkIfUserOnline) {
  1542. $use_status_in_platform = user_is_online($user_id);
  1543. $result_array['user_is_online'] = $use_status_in_platform;
  1544. $user_online_in_chat = 0;
  1545. if ($use_status_in_platform) {
  1546. $user_status = UserManager::get_extra_user_data_by_field(
  1547. $user_id,
  1548. 'user_chat_status',
  1549. false,
  1550. true
  1551. );
  1552. if ((int) $user_status['user_chat_status'] == 1) {
  1553. $user_online_in_chat = 1;
  1554. }
  1555. }
  1556. $result_array['user_is_online_in_chat'] = $user_online_in_chat;
  1557. }
  1558. if ($loadExtraData) {
  1559. $fieldValue = new ExtraFieldValue('user');
  1560. $result_array['extra'] = $fieldValue->getAllValuesForAnItem(
  1561. $user_id,
  1562. $loadOnlyVisibleExtraData
  1563. );
  1564. }
  1565. $user = _api_format_user($result_array, $showPassword, $loadAvatars);
  1566. }
  1567. if ($cacheAvailable === true) {
  1568. apcu_store($apcVar, $user, 60);
  1569. }
  1570. return $user;
  1571. }
  1572. /**
  1573. * @param int $userId
  1574. *
  1575. * @return User
  1576. */
  1577. function api_get_user_entity($userId)
  1578. {
  1579. $userId = (int) $userId;
  1580. $repo = UserManager::getRepository();
  1581. /** @var User $user */
  1582. $user = $repo->find($userId);
  1583. return $user;
  1584. }
  1585. /**
  1586. * Finds all the information about a user from username instead of user id.
  1587. *
  1588. * @param string $username
  1589. *
  1590. * @return mixed $user_info array user_id, lastname, firstname, username, email or false on error
  1591. *
  1592. * @author Yannick Warnier <yannick.warnier@beeznest.com>
  1593. */
  1594. function api_get_user_info_from_username($username = '')
  1595. {
  1596. if (empty($username)) {
  1597. return false;
  1598. }
  1599. $username = trim($username);
  1600. $sql = "SELECT * FROM ".Database::get_main_table(TABLE_MAIN_USER)."
  1601. WHERE username='".Database::escape_string($username)."'";
  1602. $result = Database::query($sql);
  1603. if (Database::num_rows($result) > 0) {
  1604. $resultArray = Database::fetch_array($result);
  1605. return _api_format_user($resultArray);
  1606. }
  1607. return false;
  1608. }
  1609. /**
  1610. * Get first user with an email.
  1611. *
  1612. * @param string $email
  1613. *
  1614. * @return array|bool
  1615. */
  1616. function api_get_user_info_from_email($email = '')
  1617. {
  1618. if (empty($email)) {
  1619. return false;
  1620. }
  1621. $sql = "SELECT * FROM ".Database::get_main_table(TABLE_MAIN_USER)."
  1622. WHERE email ='".Database::escape_string($email)."' LIMIT 1";
  1623. $result = Database::query($sql);
  1624. if (Database::num_rows($result) > 0) {
  1625. $resultArray = Database::fetch_array($result);
  1626. return _api_format_user($resultArray);
  1627. }
  1628. return false;
  1629. }
  1630. /**
  1631. * @return string
  1632. */
  1633. function api_get_course_id()
  1634. {
  1635. return Session::read('_cid', null);
  1636. }
  1637. /**
  1638. * Returns the current course id (integer).
  1639. *
  1640. * @param string $code Optional course code
  1641. *
  1642. * @return int
  1643. */
  1644. function api_get_course_int_id($code = null)
  1645. {
  1646. if (!empty($code)) {
  1647. $code = Database::escape_string($code);
  1648. $row = Database::select(
  1649. 'id',
  1650. Database::get_main_table(TABLE_MAIN_COURSE),
  1651. ['where' => ['code = ?' => [$code]]],
  1652. 'first'
  1653. );
  1654. if (is_array($row) && isset($row['id'])) {
  1655. return $row['id'];
  1656. } else {
  1657. return false;
  1658. }
  1659. }
  1660. return Session::read('_real_cid', 0);
  1661. }
  1662. /**
  1663. * Returns the current course directory.
  1664. *
  1665. * This function relies on api_get_course_info()
  1666. *
  1667. * @param string The course code - optional (takes it from session if not given)
  1668. *
  1669. * @return string The directory where the course is located inside the Chamilo "courses" directory
  1670. *
  1671. * @author Yannick Warnier <yannick.warnier@beeznest.com>
  1672. */
  1673. function api_get_course_path($course_code = null)
  1674. {
  1675. $info = !empty($course_code) ? api_get_course_info($course_code) : api_get_course_info();
  1676. return $info['path'];
  1677. }
  1678. /**
  1679. * Gets a course setting from the current course_setting table. Try always using integer values.
  1680. *
  1681. * @param string $setting_name The name of the setting we want from the table
  1682. * @param string $course_code
  1683. *
  1684. * @return mixed The value of that setting in that table. Return -1 if not found.
  1685. */
  1686. function api_get_course_setting($setting_name, $course_code = null)
  1687. {
  1688. $course_info = api_get_course_info($course_code);
  1689. $table = Database::get_course_table(TABLE_COURSE_SETTING);
  1690. $setting_name = Database::escape_string($setting_name);
  1691. if (!empty($course_info['real_id']) && !empty($setting_name)) {
  1692. $sql = "SELECT value FROM $table
  1693. WHERE c_id = {$course_info['real_id']} AND variable = '$setting_name'";
  1694. $res = Database::query($sql);
  1695. if (Database::num_rows($res) > 0) {
  1696. $row = Database::fetch_array($res);
  1697. if ($setting_name === 'email_alert_manager_on_new_quiz') {
  1698. if (!is_null($row['value'])) {
  1699. $result = explode(',', $row['value']);
  1700. $row['value'] = $result;
  1701. }
  1702. }
  1703. return $row['value'];
  1704. }
  1705. }
  1706. return -1;
  1707. }
  1708. /**
  1709. * Gets an anonymous user ID.
  1710. *
  1711. * For some tools that need tracking, like the learnpath tool, it is necessary
  1712. * to have a usable user-id to enable some kind of tracking, even if not
  1713. * perfect. An anonymous ID is taken from the users table by looking for a
  1714. * status of "6" (anonymous).
  1715. *
  1716. * @return int User ID of the anonymous user, or O if no anonymous user found
  1717. */
  1718. function api_get_anonymous_id()
  1719. {
  1720. // Find if another anon is connected now
  1721. $table = Database::get_main_table(TABLE_STATISTIC_TRACK_E_LOGIN);
  1722. $tableU = Database::get_main_table(TABLE_MAIN_USER);
  1723. $now = api_get_utc_datetime();
  1724. $ip = Database::escape_string(api_get_real_ip());
  1725. $max = (int) api_get_configuration_value('max_anonymous_users');
  1726. if ($max >= 2) {
  1727. $sql = "SELECT * FROM $table as TEL
  1728. JOIN $tableU as U
  1729. ON U.user_id = TEL.login_user_id
  1730. WHERE TEL.user_ip = '$ip'
  1731. AND U.status = ".ANONYMOUS."
  1732. AND U.user_id != 2 ";
  1733. $result = Database::query($sql);
  1734. if (empty(Database::num_rows($result))) {
  1735. $login = uniqid('anon_');
  1736. $anonList = UserManager::get_user_list(['status' => ANONYMOUS], ['registration_date ASC']);
  1737. if (count($anonList) >= $max) {
  1738. foreach ($anonList as $userToDelete) {
  1739. UserManager::delete_user($userToDelete['user_id']);
  1740. break;
  1741. }
  1742. }
  1743. $userId = UserManager::create_user(
  1744. $login,
  1745. 'anon',
  1746. ANONYMOUS,
  1747. ' anonymous@localhost',
  1748. $login,
  1749. $login
  1750. );
  1751. return $userId;
  1752. } else {
  1753. $row = Database::fetch_array($result, 'ASSOC');
  1754. return $row['user_id'];
  1755. }
  1756. }
  1757. $table = Database::get_main_table(TABLE_MAIN_USER);
  1758. $sql = "SELECT user_id
  1759. FROM $table
  1760. WHERE status = ".ANONYMOUS." ";
  1761. $res = Database::query($sql);
  1762. if (Database::num_rows($res) > 0) {
  1763. $row = Database::fetch_array($res, 'ASSOC');
  1764. return $row['user_id'];
  1765. }
  1766. // No anonymous user was found.
  1767. return 0;
  1768. }
  1769. /**
  1770. * @param string $courseCode
  1771. * @param int $sessionId
  1772. * @param int $groupId
  1773. *
  1774. * @return string
  1775. */
  1776. function api_get_cidreq_params($courseCode, $sessionId = 0, $groupId = 0)
  1777. {
  1778. $courseCode = !empty($courseCode) ? htmlspecialchars($courseCode) : '';
  1779. $sessionId = !empty($sessionId) ? (int) $sessionId : 0;
  1780. $groupId = !empty($groupId) ? (int) $groupId : 0;
  1781. $url = 'cidReq='.$courseCode;
  1782. $url .= '&id_session='.$sessionId;
  1783. $url .= '&gidReq='.$groupId;
  1784. return $url;
  1785. }
  1786. /**
  1787. * Returns the current course url part including session, group, and gradebook params.
  1788. *
  1789. * @param bool $addSessionId
  1790. * @param bool $addGroupId
  1791. * @param string $origin
  1792. *
  1793. * @return string Course & session references to add to a URL
  1794. */
  1795. function api_get_cidreq($addSessionId = true, $addGroupId = true, $origin = '')
  1796. {
  1797. $courseCode = api_get_course_id();
  1798. $url = empty($courseCode) ? '' : 'cidReq='.htmlspecialchars($courseCode);
  1799. $origin = empty($origin) ? api_get_origin() : Security::remove_XSS($origin);
  1800. if ($addSessionId) {
  1801. if (!empty($url)) {
  1802. $url .= api_get_session_id() == 0 ? '&id_session=0' : '&id_session='.api_get_session_id();
  1803. }
  1804. }
  1805. if ($addGroupId) {
  1806. if (!empty($url)) {
  1807. $url .= api_get_group_id() == 0 ? '&gidReq=0' : '&gidReq='.api_get_group_id();
  1808. }
  1809. }
  1810. if (!empty($url)) {
  1811. $url .= '&gradebook='.intval(api_is_in_gradebook());
  1812. $url .= '&origin='.$origin;
  1813. }
  1814. return $url;
  1815. }
  1816. /**
  1817. * Get if we visited a gradebook page.
  1818. *
  1819. * @return bool
  1820. */
  1821. function api_is_in_gradebook()
  1822. {
  1823. return Session::read('in_gradebook', false);
  1824. }
  1825. /**
  1826. * Set that we are in a page inside a gradebook.
  1827. */
  1828. function api_set_in_gradebook()
  1829. {
  1830. Session::write('in_gradebook', true);
  1831. }
  1832. /**
  1833. * Remove gradebook session.
  1834. */
  1835. function api_remove_in_gradebook()
  1836. {
  1837. Session::erase('in_gradebook');
  1838. }
  1839. /**
  1840. * Returns the current course info array see api_format_course_array()
  1841. * If the course_code is given, the returned array gives info about that
  1842. * particular course, if none given it gets the course info from the session.
  1843. *
  1844. * @param string $course_code
  1845. * @param bool $strict
  1846. *
  1847. * @return array
  1848. */
  1849. function api_get_course_info($course_code = null, $strict = false)
  1850. {
  1851. if (!empty($course_code)) {
  1852. $course_code = Database::escape_string($course_code);
  1853. $courseId = api_get_course_int_id($course_code);
  1854. if (empty($courseId)) {
  1855. return [];
  1856. }
  1857. $course_table = Database::get_main_table(TABLE_MAIN_COURSE);
  1858. $course_cat_table = Database::get_main_table(TABLE_MAIN_CATEGORY);
  1859. $sql = "SELECT
  1860. course.*,
  1861. course_category.code faCode,
  1862. course_category.name faName
  1863. FROM $course_table
  1864. LEFT JOIN $course_cat_table
  1865. ON course.category_code = course_category.code
  1866. WHERE course.id = $courseId";
  1867. $result = Database::query($sql);
  1868. $courseInfo = [];
  1869. if (Database::num_rows($result) > 0) {
  1870. $data = Database::fetch_array($result);
  1871. $courseInfo = api_format_course_array($data);
  1872. }
  1873. return $courseInfo;
  1874. }
  1875. global $_course;
  1876. if ($_course == '-1') {
  1877. $_course = [];
  1878. }
  1879. return $_course;
  1880. }
  1881. /**
  1882. * @param int $courseId
  1883. *
  1884. * @return \Chamilo\CoreBundle\Entity\Course
  1885. */
  1886. function api_get_course_entity($courseId = 0)
  1887. {
  1888. if (empty($courseId)) {
  1889. $courseId = api_get_course_int_id();
  1890. }
  1891. return Database::getManager()->getRepository('ChamiloCoreBundle:Course')->find($courseId);
  1892. }
  1893. /**
  1894. * @param int $id
  1895. *
  1896. * @return \Chamilo\CoreBundle\Entity\Session
  1897. */
  1898. function api_get_session_entity($id = 0)
  1899. {
  1900. if (empty($id)) {
  1901. $id = api_get_session_id();
  1902. }
  1903. return Database::getManager()->getRepository('ChamiloCoreBundle:Session')->find($id);
  1904. }
  1905. /**
  1906. * Returns the current course info array.
  1907. * Now if the course_code is given, the returned array gives info about that
  1908. * particular course, not specially the current one.
  1909. *
  1910. * @param int $id Numeric ID of the course
  1911. *
  1912. * @return array The course info as an array formatted by api_format_course_array, including category.name
  1913. */
  1914. function api_get_course_info_by_id($id = null)
  1915. {
  1916. if (!empty($id)) {
  1917. $id = (int) $id;
  1918. $course_table = Database::get_main_table(TABLE_MAIN_COURSE);
  1919. $course_cat_table = Database::get_main_table(TABLE_MAIN_CATEGORY);
  1920. $sql = "SELECT
  1921. course.*,
  1922. course_category.code faCode,
  1923. course_category.name faName
  1924. FROM $course_table
  1925. LEFT JOIN $course_cat_table
  1926. ON course.category_code = course_category.code
  1927. WHERE course.id = $id";
  1928. $result = Database::query($sql);
  1929. $_course = [];
  1930. if (Database::num_rows($result) > 0) {
  1931. $row = Database::fetch_array($result);
  1932. $_course = api_format_course_array($row);
  1933. }
  1934. return $_course;
  1935. }
  1936. global $_course;
  1937. if ($_course == '-1') {
  1938. $_course = [];
  1939. }
  1940. return $_course;
  1941. }
  1942. /**
  1943. * Reformat the course array (output by api_get_course_info()) in order, mostly,
  1944. * to switch from 'code' to 'id' in the array. This is a legacy feature and is
  1945. * now possibly causing massive confusion as a new "id" field has been added to
  1946. * the course table in 1.9.0.
  1947. *
  1948. * @param $course_data
  1949. *
  1950. * @return array
  1951. *
  1952. * @todo eradicate the false "id"=code field of the $_course array and use the int id
  1953. */
  1954. function api_format_course_array($course_data)
  1955. {
  1956. if (empty($course_data)) {
  1957. return [];
  1958. }
  1959. $_course = [];
  1960. $_course['id'] = $course_data['code'];
  1961. $_course['real_id'] = $course_data['id'];
  1962. // Added
  1963. $_course['code'] = $course_data['code'];
  1964. $_course['name'] = $course_data['title'];
  1965. $_course['title'] = $course_data['title'];
  1966. $_course['official_code'] = $course_data['visual_code'];
  1967. $_course['visual_code'] = $course_data['visual_code'];
  1968. $_course['sysCode'] = $course_data['code'];
  1969. $_course['path'] = $course_data['directory']; // Use as key in path.
  1970. $_course['directory'] = $course_data['directory'];
  1971. $_course['creation_date'] = $course_data['creation_date'];
  1972. $_course['titular'] = $course_data['tutor_name'];
  1973. $_course['language'] = $course_data['course_language'];
  1974. $_course['extLink']['url'] = $course_data['department_url'];
  1975. $_course['extLink']['name'] = $course_data['department_name'];
  1976. $_course['categoryCode'] = $course_data['faCode'];
  1977. $_course['categoryName'] = $course_data['faName'];
  1978. $_course['visibility'] = $course_data['visibility'];
  1979. $_course['subscribe_allowed'] = $course_data['subscribe'];
  1980. $_course['subscribe'] = $course_data['subscribe'];
  1981. $_course['unsubscribe'] = $course_data['unsubscribe'];
  1982. $_course['course_language'] = $course_data['course_language'];
  1983. $_course['activate_legal'] = isset($course_data['activate_legal']) ? $course_data['activate_legal'] : false;
  1984. $_course['legal'] = $course_data['legal'];
  1985. $_course['show_score'] = $course_data['show_score']; //used in the work tool
  1986. $_course['department_name'] = $course_data['department_name'];
  1987. $_course['department_url'] = $course_data['department_url'];
  1988. $courseSys = api_get_path(SYS_COURSE_PATH).$course_data['directory'];
  1989. $webCourseHome = api_get_path(WEB_COURSE_PATH).$course_data['directory'];
  1990. // Course password
  1991. $_course['registration_code'] = !empty($course_data['registration_code']) ? sha1($course_data['registration_code']) : null;
  1992. $_course['disk_quota'] = $course_data['disk_quota'];
  1993. $_course['course_public_url'] = $webCourseHome.'/index.php';
  1994. $_course['course_sys_path'] = $courseSys.'/';
  1995. if (array_key_exists('add_teachers_to_sessions_courses', $course_data)) {
  1996. $_course['add_teachers_to_sessions_courses'] = $course_data['add_teachers_to_sessions_courses'];
  1997. }
  1998. // Course image
  1999. $_course['course_image_source'] = '';
  2000. if (file_exists($courseSys.'/course-pic85x85.png')) {
  2001. $url_image = $webCourseHome.'/course-pic85x85.png';
  2002. $_course['course_image_source'] = $courseSys.'/course-pic85x85.png';
  2003. } else {
  2004. $url_image = Display::return_icon(
  2005. 'course.png',
  2006. null,
  2007. null,
  2008. ICON_SIZE_BIG,
  2009. null,
  2010. true,
  2011. false
  2012. );
  2013. }
  2014. $_course['course_image'] = $url_image;
  2015. // Course large image
  2016. $_course['course_image_large_source'] = '';
  2017. if (file_exists($courseSys.'/course-pic.png')) {
  2018. $url_image = $webCourseHome.'/course-pic.png';
  2019. $_course['course_image_large_source'] = $courseSys.'/course-pic.png';
  2020. } else {
  2021. $url_image = Display::return_icon(
  2022. 'session_default.png',
  2023. null,
  2024. null,
  2025. null,
  2026. null,
  2027. true,
  2028. true
  2029. );
  2030. }
  2031. $_course['course_image_large'] = $url_image;
  2032. return $_course;
  2033. }
  2034. /**
  2035. * Returns a difficult to guess password.
  2036. *
  2037. * @param int $length the length of the password
  2038. *
  2039. * @return string the generated password
  2040. */
  2041. function api_generate_password($length = 8)
  2042. {
  2043. if ($length < 2) {
  2044. $length = 2;
  2045. }
  2046. $charactersLowerCase = 'abcdefghijkmnopqrstuvwxyz';
  2047. $charactersUpperCase = 'ABCDEFGHJKLMNPQRSTUVWXYZ';
  2048. $minNumbers = 2;
  2049. $length = $length - $minNumbers;
  2050. $minLowerCase = round($length / 2);
  2051. $minUpperCase = $length - $minLowerCase;
  2052. $password = '';
  2053. $passwordRequirements = api_get_configuration_value('password_requirements');
  2054. $factory = new RandomLib\Factory();
  2055. $generator = $factory->getGenerator(new SecurityLib\Strength(SecurityLib\Strength::MEDIUM));
  2056. if (!empty($passwordRequirements)) {
  2057. $length = $passwordRequirements['min']['length'];
  2058. $minNumbers = $passwordRequirements['min']['numeric'];
  2059. $minLowerCase = $passwordRequirements['min']['lowercase'];
  2060. $minUpperCase = $passwordRequirements['min']['uppercase'];
  2061. $rest = $length - $minNumbers - $minLowerCase - $minUpperCase;
  2062. // Add the rest to fill the length requirement
  2063. if ($rest > 0) {
  2064. $password .= $generator->generateString($rest, $charactersLowerCase.$charactersUpperCase);
  2065. }
  2066. }
  2067. // Min digits default 2
  2068. for ($i = 0; $i < $minNumbers; $i++) {
  2069. $password .= $generator->generateInt(2, 9);
  2070. }
  2071. // Min lowercase
  2072. $password .= $generator->generateString($minLowerCase, $charactersLowerCase);
  2073. // Min uppercase
  2074. $password .= $generator->generateString($minUpperCase, $charactersUpperCase);
  2075. $password = str_shuffle($password);
  2076. return $password;
  2077. }
  2078. /**
  2079. * Checks a password to see wether it is OK to use.
  2080. *
  2081. * @param string $password
  2082. *
  2083. * @return bool if the password is acceptable, false otherwise
  2084. * Notes about what a password "OK to use" is:
  2085. * 1. The password should be at least 5 characters long.
  2086. * 2. Only English letters (uppercase or lowercase, it doesn't matter) and digits are allowed.
  2087. * 3. The password should contain at least 3 letters.
  2088. * 4. It should contain at least 2 digits.
  2089. * Settings will change if the configuration value is set: password_requirements
  2090. */
  2091. function api_check_password($password)
  2092. {
  2093. $passwordRequirements = Security::getPasswordRequirements();
  2094. $minLength = $passwordRequirements['min']['length'];
  2095. $minNumbers = $passwordRequirements['min']['numeric'];
  2096. // Optional
  2097. $minLowerCase = $passwordRequirements['min']['lowercase'];
  2098. $minUpperCase = $passwordRequirements['min']['uppercase'];
  2099. $minLetters = $minLowerCase + $minUpperCase;
  2100. $passwordLength = api_strlen($password);
  2101. $conditions = [
  2102. 'min_length' => $passwordLength >= $minLength,
  2103. ];
  2104. $digits = 0;
  2105. $lowerCase = 0;
  2106. $upperCase = 0;
  2107. for ($i = 0; $i < $passwordLength; $i++) {
  2108. $currentCharacterCode = api_ord(api_substr($password, $i, 1));
  2109. if ($currentCharacterCode >= 65 && $currentCharacterCode <= 90) {
  2110. $upperCase++;
  2111. }
  2112. if ($currentCharacterCode >= 97 && $currentCharacterCode <= 122) {
  2113. $lowerCase++;
  2114. }
  2115. if ($currentCharacterCode >= 48 && $currentCharacterCode <= 57) {
  2116. $digits++;
  2117. }
  2118. }
  2119. // Min number of digits
  2120. $conditions['min_numeric'] = $digits >= $minNumbers;
  2121. if (!empty($minUpperCase)) {
  2122. // Uppercase
  2123. $conditions['min_uppercase'] = $upperCase >= $minUpperCase;
  2124. }
  2125. if (!empty($minLowerCase)) {
  2126. // Lowercase
  2127. $conditions['min_lowercase'] = $upperCase >= $minLowerCase;
  2128. }
  2129. // Min letters
  2130. $letters = $upperCase + $lowerCase;
  2131. $conditions['min_letters'] = $letters >= $minLetters;
  2132. $isPasswordOk = true;
  2133. foreach ($conditions as $condition) {
  2134. if ($condition === false) {
  2135. $isPasswordOk = false;
  2136. break;
  2137. }
  2138. }
  2139. if ($isPasswordOk === false) {
  2140. $output = get_lang('NewPasswordRequirementsNotMatched').'<br />';
  2141. $output .= Security::getPasswordRequirementsToString($conditions);
  2142. Display::addFlash(Display::return_message($output, 'warning', false));
  2143. }
  2144. return $isPasswordOk;
  2145. }
  2146. /**
  2147. * Clears the user ID from the session if it was the anonymous user. Generally
  2148. * used on out-of-tools pages to remove a user ID that could otherwise be used
  2149. * in the wrong context.
  2150. * This function is to be used in conjunction with the api_set_anonymous()
  2151. * function to simulate the user existence in case of an anonymous visit.
  2152. *
  2153. * @param bool database check switch - passed to api_is_anonymous()
  2154. *
  2155. * @return bool true if succesfully unregistered, false if not anonymous
  2156. */
  2157. function api_clear_anonymous($db_check = false)
  2158. {
  2159. global $_user;
  2160. if (api_is_anonymous($_user['user_id'], $db_check)) {
  2161. unset($_user['user_id']);
  2162. Session::erase('_uid');
  2163. return true;
  2164. }
  2165. return false;
  2166. }
  2167. /**
  2168. * Returns the status string corresponding to the status code.
  2169. *
  2170. * @author Noel Dieschburg
  2171. *
  2172. * @param the int status code
  2173. *
  2174. * @return string
  2175. */
  2176. function get_status_from_code($status_code)
  2177. {
  2178. switch ($status_code) {
  2179. case STUDENT:
  2180. return get_lang('Student', '');
  2181. case COURSEMANAGER:
  2182. return get_lang('Teacher', '');
  2183. case SESSIONADMIN:
  2184. return get_lang('SessionsAdmin', '');
  2185. case DRH:
  2186. return get_lang('Drh', '');
  2187. }
  2188. }
  2189. /**
  2190. * Sets the current user as anonymous if it hasn't been identified yet. This
  2191. * function should be used inside a tool only. The function api_clear_anonymous()
  2192. * acts in the opposite direction by clearing the anonymous user's data every
  2193. * time we get on a course homepage or on a neutral page (index, admin, my space).
  2194. *
  2195. * @return bool true if set user as anonymous, false if user was already logged in or anonymous id could not be found
  2196. */
  2197. function api_set_anonymous()
  2198. {
  2199. global $_user;
  2200. if (!empty($_user['user_id'])) {
  2201. return false;
  2202. }
  2203. $user_id = api_get_anonymous_id();
  2204. if ($user_id == 0) {
  2205. return false;
  2206. }
  2207. if (isset($_user['is_anonymous'])) {
  2208. return false;
  2209. }
  2210. Session::erase('_user');
  2211. $_user['user_id'] = $user_id;
  2212. $_user['is_anonymous'] = true;
  2213. $GLOBALS['_user'] = $_user;
  2214. Session::write('_user', $_user);
  2215. return true;
  2216. }
  2217. /**
  2218. * Gets the current Chamilo (not PHP/cookie) session ID.
  2219. *
  2220. * @return int O if no active session, the session ID otherwise
  2221. */
  2222. function api_get_session_id()
  2223. {
  2224. return (int) Session::read('id_session', 0);
  2225. }
  2226. /**
  2227. * Gets the current Chamilo (not social network) group ID.
  2228. *
  2229. * @return int O if no active session, the session ID otherwise
  2230. */
  2231. function api_get_group_id()
  2232. {
  2233. return Session::read('_gid', 0);
  2234. }
  2235. /**
  2236. * Gets the current or given session name.
  2237. *
  2238. * @param int Session ID (optional)
  2239. *
  2240. * @return string The session name, or null if not found
  2241. */
  2242. function api_get_session_name($session_id = 0)
  2243. {
  2244. if (empty($session_id)) {
  2245. $session_id = api_get_session_id();
  2246. if (empty($session_id)) {
  2247. return null;
  2248. }
  2249. }
  2250. $t = Database::get_main_table(TABLE_MAIN_SESSION);
  2251. $s = "SELECT name FROM $t WHERE id = ".(int) $session_id;
  2252. $r = Database::query($s);
  2253. $c = Database::num_rows($r);
  2254. if ($c > 0) {
  2255. //technically, there can be only one, but anyway we take the first
  2256. $rec = Database::fetch_array($r);
  2257. return $rec['name'];
  2258. }
  2259. return null;
  2260. }
  2261. /**
  2262. * Gets the session info by id.
  2263. *
  2264. * @param int $id Session ID
  2265. *
  2266. * @return array information of the session
  2267. */
  2268. function api_get_session_info($id)
  2269. {
  2270. return SessionManager::fetch($id);
  2271. }
  2272. /**
  2273. * Gets the session visibility by session id.
  2274. *
  2275. * @param int $session_id
  2276. * @param int $courseId
  2277. * @param bool $ignore_visibility_for_admins
  2278. *
  2279. * @return int
  2280. * 0 = session still available,
  2281. * SESSION_VISIBLE_READ_ONLY = 1,
  2282. * SESSION_VISIBLE = 2,
  2283. * SESSION_INVISIBLE = 3
  2284. */
  2285. function api_get_session_visibility(
  2286. $session_id,
  2287. $courseId = null,
  2288. $ignore_visibility_for_admins = true
  2289. ) {
  2290. if (api_is_platform_admin()) {
  2291. if ($ignore_visibility_for_admins) {
  2292. return SESSION_AVAILABLE;
  2293. }
  2294. }
  2295. $now = time();
  2296. if (empty($session_id)) {
  2297. return 0; // Means that the session is still available.
  2298. }
  2299. $session_id = (int) $session_id;
  2300. $tbl_session = Database::get_main_table(TABLE_MAIN_SESSION);
  2301. $result = Database::query("SELECT * FROM $tbl_session WHERE id = $session_id");
  2302. if (Database::num_rows($result) <= 0) {
  2303. return SESSION_INVISIBLE;
  2304. }
  2305. $row = Database::fetch_array($result, 'ASSOC');
  2306. $visibility = $original_visibility = $row['visibility'];
  2307. // I don't care the session visibility.
  2308. if (empty($row['access_start_date']) && empty($row['access_end_date'])) {
  2309. // Session duration per student.
  2310. if (isset($row['duration']) && !empty($row['duration'])) {
  2311. $duration = $row['duration'] * 24 * 60 * 60;
  2312. $courseAccess = CourseManager::getFirstCourseAccessPerSessionAndUser($session_id, api_get_user_id());
  2313. // If there is a session duration but there is no previous
  2314. // access by the user, then the session is still available
  2315. if (count($courseAccess) == 0) {
  2316. return SESSION_AVAILABLE;
  2317. }
  2318. $currentTime = time();
  2319. $firstAccess = isset($courseAccess['login_course_date'])
  2320. ? api_strtotime($courseAccess['login_course_date'], 'UTC')
  2321. : 0;
  2322. $userDurationData = SessionManager::getUserSession(
  2323. api_get_user_id(),
  2324. $session_id
  2325. );
  2326. $userDuration = isset($userDurationData['duration'])
  2327. ? (intval($userDurationData['duration']) * 24 * 60 * 60)
  2328. : 0;
  2329. $totalDuration = $firstAccess + $duration + $userDuration;
  2330. return $totalDuration > $currentTime ? SESSION_AVAILABLE : SESSION_VISIBLE_READ_ONLY;
  2331. }
  2332. return SESSION_AVAILABLE;
  2333. }
  2334. // If start date was set.
  2335. if (!empty($row['access_start_date'])) {
  2336. $visibility = $now > api_strtotime($row['access_start_date'], 'UTC') ? SESSION_AVAILABLE : SESSION_INVISIBLE;
  2337. }
  2338. // If the end date was set.
  2339. if (!empty($row['access_end_date'])) {
  2340. // Only if date_start said that it was ok
  2341. if ($visibility === SESSION_AVAILABLE) {
  2342. $visibility = $now < api_strtotime($row['access_end_date'], 'UTC')
  2343. ? SESSION_AVAILABLE // Date still available
  2344. : $row['visibility']; // Session ends
  2345. }
  2346. }
  2347. // If I'm a coach the visibility can change in my favor depending in the coach dates.
  2348. $isCoach = api_is_coach($session_id, $courseId);
  2349. if ($isCoach) {
  2350. // Test start date.
  2351. if (!empty($row['coach_access_start_date'])) {
  2352. $start = api_strtotime($row['coach_access_start_date'], 'UTC');
  2353. $visibility = $start < $now ? SESSION_AVAILABLE : SESSION_INVISIBLE;
  2354. }
  2355. // Test end date.
  2356. if (!empty($row['coach_access_end_date'])) {
  2357. if ($visibility === SESSION_AVAILABLE) {
  2358. $endDateCoach = api_strtotime($row['coach_access_end_date'], 'UTC');
  2359. $visibility = $endDateCoach >= $now ? SESSION_AVAILABLE : $row['visibility'];
  2360. }
  2361. }
  2362. }
  2363. return $visibility;
  2364. }
  2365. /**
  2366. * This function returns a (star) session icon if the session is not null and
  2367. * the user is not a student.
  2368. *
  2369. * @param int $sessionId
  2370. * @param int $statusId User status id - if 5 (student), will return empty
  2371. *
  2372. * @return string Session icon
  2373. */
  2374. function api_get_session_image($sessionId, $statusId)
  2375. {
  2376. $sessionId = (int) $sessionId;
  2377. $image = '';
  2378. if ($statusId != STUDENT) {
  2379. // Check whether is not a student
  2380. if ($sessionId > 0) {
  2381. $image = '&nbsp;&nbsp;'.Display::return_icon(
  2382. 'star.png',
  2383. get_lang('SessionSpecificResource'),
  2384. ['align' => 'absmiddle'],
  2385. ICON_SIZE_SMALL
  2386. );
  2387. }
  2388. }
  2389. return $image;
  2390. }
  2391. /**
  2392. * This function add an additional condition according to the session of the course.
  2393. *
  2394. * @param int $session_id session id
  2395. * @param bool $and optional, true if more than one condition false if the only condition in the query
  2396. * @param bool $with_base_content optional, true to accept content with session=0 as well,
  2397. * false for strict session condition
  2398. * @param string $session_field
  2399. *
  2400. * @return string condition of the session
  2401. */
  2402. function api_get_session_condition(
  2403. $session_id,
  2404. $and = true,
  2405. $with_base_content = false,
  2406. $session_field = 'session_id'
  2407. ) {
  2408. $session_id = (int) $session_id;
  2409. if (empty($session_field)) {
  2410. $session_field = 'session_id';
  2411. }
  2412. // Condition to show resources by session
  2413. $condition_add = $and ? ' AND ' : ' WHERE ';
  2414. if ($with_base_content) {
  2415. $condition_session = $condition_add." ( $session_field = $session_id OR $session_field = 0 OR $session_field IS NULL) ";
  2416. } else {
  2417. if (empty($session_id)) {
  2418. $condition_session = $condition_add." ($session_field = $session_id OR $session_field IS NULL)";
  2419. } else {
  2420. $condition_session = $condition_add." $session_field = $session_id ";
  2421. }
  2422. }
  2423. return $condition_session;
  2424. }
  2425. /**
  2426. * Returns the value of a setting from the web-adjustable admin config settings.
  2427. *
  2428. * WARNING true/false are stored as string, so when comparing you need to check e.g.
  2429. * if (api_get_setting('show_navigation_menu') == 'true') //CORRECT
  2430. * instead of
  2431. * if (api_get_setting('show_navigation_menu') == true) //INCORRECT
  2432. *
  2433. * @param string $variable The variable name
  2434. * @param string $key The subkey (sub-variable) if any. Defaults to NULL
  2435. *
  2436. * @return string
  2437. *
  2438. * @author René Haentjens
  2439. * @author Bart Mollet
  2440. */
  2441. function api_get_setting($variable, $key = null)
  2442. {
  2443. global $_setting;
  2444. if ($variable == 'header_extra_content') {
  2445. $filename = api_get_home_path().'header_extra_content.txt';
  2446. if (file_exists($filename)) {
  2447. $value = file_get_contents($filename);
  2448. return $value;
  2449. } else {
  2450. return '';
  2451. }
  2452. }
  2453. if ($variable == 'footer_extra_content') {
  2454. $filename = api_get_home_path().'footer_extra_content.txt';
  2455. if (file_exists($filename)) {
  2456. $value = file_get_contents($filename);
  2457. return $value;
  2458. } else {
  2459. return '';
  2460. }
  2461. }
  2462. $value = null;
  2463. if (is_null($key)) {
  2464. $value = ((isset($_setting[$variable]) && $_setting[$variable] != '') ? $_setting[$variable] : null);
  2465. } else {
  2466. if (isset($_setting[$variable][$key])) {
  2467. $value = $_setting[$variable][$key];
  2468. }
  2469. }
  2470. return $value;
  2471. }
  2472. /**
  2473. * @param string $plugin
  2474. * @param string $variable
  2475. *
  2476. * @return string
  2477. */
  2478. function api_get_plugin_setting($plugin, $variable)
  2479. {
  2480. $variableName = $plugin.'_'.$variable;
  2481. $result = api_get_setting($variableName);
  2482. if (isset($result[$plugin])) {
  2483. $value = $result[$plugin];
  2484. $unserialized = UnserializeApi::unserialize('not_allowed_classes', $value, true);
  2485. if (false !== $unserialized) {
  2486. $value = $unserialized;
  2487. }
  2488. return $value;
  2489. }
  2490. return null;
  2491. }
  2492. /**
  2493. * Returns the value of a setting from the web-adjustable admin config settings.
  2494. */
  2495. function api_get_settings_params($params)
  2496. {
  2497. $table = Database::get_main_table(TABLE_MAIN_SETTINGS_CURRENT);
  2498. $result = Database::select('*', $table, ['where' => $params]);
  2499. return $result;
  2500. }
  2501. /**
  2502. * @param array $params example: [id = ? => '1']
  2503. *
  2504. * @return array
  2505. */
  2506. function api_get_settings_params_simple($params)
  2507. {
  2508. $table = Database::get_main_table(TABLE_MAIN_SETTINGS_CURRENT);
  2509. $result = Database::select('*', $table, ['where' => $params], 'one');
  2510. return $result;
  2511. }
  2512. /**
  2513. * Returns the value of a setting from the web-adjustable admin config settings.
  2514. */
  2515. function api_delete_settings_params($params)
  2516. {
  2517. $table = Database::get_main_table(TABLE_MAIN_SETTINGS_CURRENT);
  2518. $result = Database::delete($table, $params);
  2519. return $result;
  2520. }
  2521. /**
  2522. * Returns an escaped version of $_SERVER['PHP_SELF'] to avoid XSS injection.
  2523. *
  2524. * @return string Escaped version of $_SERVER['PHP_SELF']
  2525. */
  2526. function api_get_self()
  2527. {
  2528. return htmlentities($_SERVER['PHP_SELF']);
  2529. }
  2530. /* USER PERMISSIONS */
  2531. /**
  2532. * Checks whether current user is a platform administrator.
  2533. *
  2534. * @param bool $allowSessionAdmins Whether session admins should be considered admins or not
  2535. * @param bool $allowDrh Whether HR directors should be considered admins or not
  2536. *
  2537. * @return bool true if the user has platform admin rights,
  2538. * false otherwise
  2539. *
  2540. * @see usermanager::is_admin(user_id) for a user-id specific function
  2541. */
  2542. function api_is_platform_admin($allowSessionAdmins = false, $allowDrh = false)
  2543. {
  2544. $isAdmin = Session::read('is_platformAdmin');
  2545. if ($isAdmin) {
  2546. return true;
  2547. }
  2548. $user = api_get_user_info();
  2549. return
  2550. isset($user['status']) &&
  2551. (
  2552. ($allowSessionAdmins && $user['status'] == SESSIONADMIN) ||
  2553. ($allowDrh && $user['status'] == DRH)
  2554. );
  2555. }
  2556. /**
  2557. * Checks whether the user given as user id is in the admin table.
  2558. *
  2559. * @param int $user_id If none provided, will use current user
  2560. * @param int $url URL ID. If provided, also check if the user is active on given URL
  2561. *
  2562. * @return bool True if the user is admin, false otherwise
  2563. */
  2564. function api_is_platform_admin_by_id($user_id = null, $url = null)
  2565. {
  2566. $user_id = (int) $user_id;
  2567. if (empty($user_id)) {
  2568. $user_id = api_get_user_id();
  2569. }
  2570. $admin_table = Database::get_main_table(TABLE_MAIN_ADMIN);
  2571. $sql = "SELECT * FROM $admin_table WHERE user_id = $user_id";
  2572. $res = Database::query($sql);
  2573. $is_admin = Database::num_rows($res) === 1;
  2574. if (!$is_admin || !isset($url)) {
  2575. return $is_admin;
  2576. }
  2577. // We get here only if $url is set
  2578. $url = (int) $url;
  2579. $url_user_table = Database::get_main_table(TABLE_MAIN_ACCESS_URL_REL_USER);
  2580. $sql = "SELECT * FROM $url_user_table
  2581. WHERE access_url_id = $url AND user_id = $user_id";
  2582. $res = Database::query($sql);
  2583. $result = Database::num_rows($res) === 1;
  2584. return $result;
  2585. }
  2586. /**
  2587. * Returns the user's numeric status ID from the users table.
  2588. *
  2589. * @param int $user_id If none provided, will use current user
  2590. *
  2591. * @return int User's status (1 for teacher, 5 for student, etc)
  2592. */
  2593. function api_get_user_status($user_id = null)
  2594. {
  2595. $user_id = (int) $user_id;
  2596. if (empty($user_id)) {
  2597. $user_id = api_get_user_id();
  2598. }
  2599. $table = Database::get_main_table(TABLE_MAIN_USER);
  2600. $sql = "SELECT status FROM $table WHERE user_id = $user_id ";
  2601. $result = Database::query($sql);
  2602. $status = null;
  2603. if (Database::num_rows($result)) {
  2604. $row = Database::fetch_array($result);
  2605. $status = $row['status'];
  2606. }
  2607. return $status;
  2608. }
  2609. /**
  2610. * Checks whether current user is allowed to create courses.
  2611. *
  2612. * @return bool true if the user has course creation rights,
  2613. * false otherwise
  2614. */
  2615. function api_is_allowed_to_create_course()
  2616. {
  2617. if (api_is_platform_admin()) {
  2618. return true;
  2619. }
  2620. // Teachers can only create courses
  2621. if (api_is_teacher()) {
  2622. if (api_get_setting('allow_users_to_create_courses') === 'true') {
  2623. return true;
  2624. } else {
  2625. return false;
  2626. }
  2627. }
  2628. return Session::read('is_allowedCreateCourse');
  2629. }
  2630. /**
  2631. * Checks whether the current user is a course administrator.
  2632. *
  2633. * @return bool True if current user is a course administrator
  2634. */
  2635. function api_is_course_admin()
  2636. {
  2637. if (api_is_platform_admin()) {
  2638. return true;
  2639. }
  2640. return Session::read('is_courseAdmin');
  2641. }
  2642. /**
  2643. * Checks whether the current user is a course coach
  2644. * Based on the presence of user in session.id_coach (session general coach).
  2645. *
  2646. * @return bool True if current user is a course coach
  2647. */
  2648. function api_is_session_general_coach()
  2649. {
  2650. return Session::read('is_session_general_coach');
  2651. }
  2652. /**
  2653. * Checks whether the current user is a course tutor
  2654. * Based on the presence of user in session_rel_course_rel_user.user_id with status = 2.
  2655. *
  2656. * @return bool True if current user is a course tutor
  2657. */
  2658. function api_is_course_tutor()
  2659. {
  2660. return Session::read('is_courseTutor');
  2661. }
  2662. /**
  2663. * @param int $user_id
  2664. * @param int $courseId
  2665. * @param int $session_id
  2666. *
  2667. * @return bool
  2668. */
  2669. function api_is_course_session_coach($user_id, $courseId, $session_id)
  2670. {
  2671. $session_table = Database::get_main_table(TABLE_MAIN_SESSION);
  2672. $session_rel_course_rel_user_table = Database::get_main_table(TABLE_MAIN_SESSION_COURSE_USER);
  2673. $user_id = (int) $user_id;
  2674. $session_id = (int) $session_id;
  2675. $courseId = (int) $courseId;
  2676. $sql = "SELECT DISTINCT session.id
  2677. FROM $session_table
  2678. INNER JOIN $session_rel_course_rel_user_table session_rc_ru
  2679. ON session.id = session_rc_ru.session_id
  2680. WHERE
  2681. session_rc_ru.user_id = '".$user_id."' AND
  2682. session_rc_ru.c_id = '$courseId' AND
  2683. session_rc_ru.status = 2 AND
  2684. session_rc_ru.session_id = '$session_id'";
  2685. $result = Database::query($sql);
  2686. return Database::num_rows($result) > 0;
  2687. }
  2688. /**
  2689. * Checks whether the current user is a course or session coach.
  2690. *
  2691. * @param int $session_id
  2692. * @param int $courseId
  2693. * @param bool Check whether we are in student view and, if we are, return false
  2694. *
  2695. * @return bool True if current user is a course or session coach
  2696. */
  2697. function api_is_coach($session_id = 0, $courseId = null, $check_student_view = true)
  2698. {
  2699. $userId = api_get_user_id();
  2700. if (!empty($session_id)) {
  2701. $session_id = (int) $session_id;
  2702. } else {
  2703. $session_id = api_get_session_id();
  2704. }
  2705. // The student preview was on
  2706. if ($check_student_view && api_is_student_view_active()) {
  2707. return false;
  2708. }
  2709. if (!empty($courseId)) {
  2710. $courseId = (int) $courseId;
  2711. } else {
  2712. $courseId = api_get_course_int_id();
  2713. }
  2714. $session_table = Database::get_main_table(TABLE_MAIN_SESSION);
  2715. $session_rel_course_rel_user_table = Database::get_main_table(TABLE_MAIN_SESSION_COURSE_USER);
  2716. $sessionIsCoach = [];
  2717. if (!empty($courseId)) {
  2718. $sql = "SELECT DISTINCT s.id, name, access_start_date, access_end_date
  2719. FROM $session_table s
  2720. INNER JOIN $session_rel_course_rel_user_table session_rc_ru
  2721. ON session_rc_ru.session_id = s.id AND session_rc_ru.user_id = '".$userId."'
  2722. WHERE
  2723. session_rc_ru.c_id = '$courseId' AND
  2724. session_rc_ru.status = 2 AND
  2725. session_rc_ru.session_id = '$session_id'";
  2726. $result = Database::query($sql);
  2727. $sessionIsCoach = Database::store_result($result);
  2728. }
  2729. if (!empty($session_id)) {
  2730. $sql = "SELECT DISTINCT id, name, access_start_date, access_end_date
  2731. FROM $session_table
  2732. WHERE session.id_coach = $userId AND id = $session_id
  2733. ORDER BY access_start_date, access_end_date, name";
  2734. $result = Database::query($sql);
  2735. if (!empty($sessionIsCoach)) {
  2736. $sessionIsCoach = array_merge(
  2737. $sessionIsCoach,
  2738. Database::store_result($result)
  2739. );
  2740. } else {
  2741. $sessionIsCoach = Database::store_result($result);
  2742. }
  2743. }
  2744. return count($sessionIsCoach) > 0;
  2745. }
  2746. /**
  2747. * Checks whether the current user is a session administrator.
  2748. *
  2749. * @return bool True if current user is a course administrator
  2750. */
  2751. function api_is_session_admin()
  2752. {
  2753. $user = api_get_user_info();
  2754. return isset($user['status']) && $user['status'] == SESSIONADMIN;
  2755. }
  2756. /**
  2757. * Checks whether the current user is a human resources manager.
  2758. *
  2759. * @return bool True if current user is a human resources manager
  2760. */
  2761. function api_is_drh()
  2762. {
  2763. $user = api_get_user_info();
  2764. return isset($user['status']) && $user['status'] == DRH;
  2765. }
  2766. /**
  2767. * Checks whether the current user is a student.
  2768. *
  2769. * @return bool True if current user is a human resources manager
  2770. */
  2771. function api_is_student()
  2772. {
  2773. $user = api_get_user_info();
  2774. return isset($user['status']) && $user['status'] == STUDENT;
  2775. }
  2776. /**
  2777. * Checks whether the current user has the status 'teacher'.
  2778. *
  2779. * @return bool True if current user is a human resources manager
  2780. */
  2781. function api_is_teacher()
  2782. {
  2783. $user = api_get_user_info();
  2784. return isset($user['status']) && $user['status'] == COURSEMANAGER;
  2785. }
  2786. /**
  2787. * Checks whether the current user is a invited user.
  2788. *
  2789. * @return bool
  2790. */
  2791. function api_is_invitee()
  2792. {
  2793. $user = api_get_user_info();
  2794. return isset($user['status']) && $user['status'] == INVITEE;
  2795. }
  2796. /**
  2797. * This function checks whether a session is assigned into a category.
  2798. *
  2799. * @param int - session id
  2800. * @param string - category name
  2801. *
  2802. * @return bool - true if is found, otherwise false
  2803. */
  2804. function api_is_session_in_category($session_id, $category_name)
  2805. {
  2806. $session_id = (int) $session_id;
  2807. $category_name = Database::escape_string($category_name);
  2808. $tbl_session = Database::get_main_table(TABLE_MAIN_SESSION);
  2809. $tbl_session_category = Database::get_main_table(TABLE_MAIN_SESSION_CATEGORY);
  2810. $sql = "SELECT 1
  2811. FROM $tbl_session
  2812. WHERE $session_id IN (
  2813. SELECT s.id FROM $tbl_session s, $tbl_session_category sc
  2814. WHERE
  2815. s.session_category_id = sc.id AND
  2816. sc.name LIKE '%$category_name'
  2817. )";
  2818. $rs = Database::query($sql);
  2819. if (Database::num_rows($rs) > 0) {
  2820. return true;
  2821. } else {
  2822. return false;
  2823. }
  2824. }
  2825. /**
  2826. * Displays the title of a tool.
  2827. * Normal use: parameter is a string:
  2828. * api_display_tool_title("My Tool").
  2829. *
  2830. * Optionally, there can be a subtitle below
  2831. * the normal title, and / or a supra title above the normal title.
  2832. *
  2833. * e.g. supra title:
  2834. * group
  2835. * GROUP PROPERTIES
  2836. *
  2837. * e.g. subtitle:
  2838. * AGENDA
  2839. * calender & events tool
  2840. *
  2841. * @author Hugues Peeters <hugues.peeters@claroline.net>
  2842. *
  2843. * @param mixed $title_element - it could either be a string or an array
  2844. * containing 'supraTitle', 'mainTitle',
  2845. * 'subTitle'
  2846. */
  2847. function api_display_tool_title($title_element)
  2848. {
  2849. if (is_string($title_element)) {
  2850. $tit = $title_element;
  2851. unset($title_element);
  2852. $title_element['mainTitle'] = $tit;
  2853. }
  2854. echo '<h3>';
  2855. if (!empty($title_element['supraTitle'])) {
  2856. echo '<small>'.$title_element['supraTitle'].'</small><br />';
  2857. }
  2858. if (!empty($title_element['mainTitle'])) {
  2859. echo $title_element['mainTitle'];
  2860. }
  2861. if (!empty($title_element['subTitle'])) {
  2862. echo '<br /><small>'.$title_element['subTitle'].'</small>';
  2863. }
  2864. echo '</h3>';
  2865. }
  2866. /**
  2867. * Displays options for switching between student view and course manager view.
  2868. *
  2869. * Changes in version 1.2 (Patrick Cool)
  2870. * Student view switch now behaves as a real switch. It maintains its current state until the state
  2871. * is changed explicitly
  2872. *
  2873. * Changes in version 1.1 (Patrick Cool)
  2874. * student view now works correctly in subfolders of the document tool
  2875. * student view works correctly in the new links tool
  2876. *
  2877. * Example code for using this in your tools:
  2878. * //if ($is_courseAdmin && api_get_setting('student_view_enabled') == 'true') {
  2879. * // display_tool_view_option($isStudentView);
  2880. * //}
  2881. * //and in later sections, use api_is_allowed_to_edit()
  2882. *
  2883. * @author Roan Embrechts
  2884. * @author Patrick Cool
  2885. * @author Julio Montoya, changes added in Chamilo
  2886. *
  2887. * @version 1.2
  2888. *
  2889. * @todo rewrite code so it is easier to understand
  2890. */
  2891. function api_display_tool_view_option()
  2892. {
  2893. if (api_get_setting('student_view_enabled') != 'true') {
  2894. return '';
  2895. }
  2896. $sourceurl = '';
  2897. $is_framed = false;
  2898. // Exceptions apply for all multi-frames pages
  2899. if (strpos($_SERVER['REQUEST_URI'], 'chat/chat_banner.php') !== false) {
  2900. // The chat is a multiframe bit that doesn't work too well with the student_view, so do not show the link
  2901. return '';
  2902. }
  2903. // Uncomment to remove student view link from document view page
  2904. if (strpos($_SERVER['REQUEST_URI'], 'lp/lp_header.php') !== false) {
  2905. if (empty($_GET['lp_id'])) {
  2906. return '';
  2907. }
  2908. $sourceurl = substr($_SERVER['REQUEST_URI'], 0, strpos($_SERVER['REQUEST_URI'], '?'));
  2909. $sourceurl = str_replace(
  2910. 'lp/lp_header.php',
  2911. 'lp/lp_controller.php?'.api_get_cidreq().'&action=view&lp_id='.intval($_GET['lp_id']).'&isStudentView='.($_SESSION['studentview'] == 'studentview' ? 'false' : 'true'),
  2912. $sourceurl
  2913. );
  2914. //showinframes doesn't handle student view anyway...
  2915. //return '';
  2916. $is_framed = true;
  2917. }
  2918. // Check whether the $_SERVER['REQUEST_URI'] contains already url parameters (thus a questionmark)
  2919. if (!$is_framed) {
  2920. if (strpos($_SERVER['REQUEST_URI'], '?') === false) {
  2921. $sourceurl = api_get_self().'?'.api_get_cidreq();
  2922. } else {
  2923. $sourceurl = $_SERVER['REQUEST_URI'];
  2924. }
  2925. }
  2926. $output_string = '';
  2927. if (!empty($_SESSION['studentview'])) {
  2928. if ($_SESSION['studentview'] == 'studentview') {
  2929. // We have to remove the isStudentView=true from the $sourceurl
  2930. $sourceurl = str_replace('&isStudentView=true', '', $sourceurl);
  2931. $sourceurl = str_replace('&isStudentView=false', '', $sourceurl);
  2932. $output_string .= '<a class="btn btn-primary btn-sm" href="'.$sourceurl.'&isStudentView=false" target="_self">'.
  2933. Display::returnFontAwesomeIcon('eye').' '.get_lang('SwitchToTeacherView').'</a>';
  2934. } elseif ($_SESSION['studentview'] == 'teacherview') {
  2935. // Switching to teacherview
  2936. $sourceurl = str_replace('&isStudentView=true', '', $sourceurl);
  2937. $sourceurl = str_replace('&isStudentView=false', '', $sourceurl);
  2938. $output_string .= '<a class="btn btn-default btn-sm" href="'.$sourceurl.'&isStudentView=true" target="_self">'.
  2939. Display::returnFontAwesomeIcon('eye').' '.get_lang('SwitchToStudentView').'</a>';
  2940. }
  2941. } else {
  2942. $output_string .= '<a class="btn btn-default btn-sm" href="'.$sourceurl.'&isStudentView=true" target="_self">'.
  2943. Display::returnFontAwesomeIcon('eye').' '.get_lang('SwitchToStudentView').'</a>';
  2944. }
  2945. $output_string = Security::remove_XSS($output_string);
  2946. $html = Display::tag('div', $output_string, ['class' => 'view-options']);
  2947. return $html;
  2948. }
  2949. // TODO: This is for the permission section.
  2950. /**
  2951. * Function that removes the need to directly use is_courseAdmin global in
  2952. * tool scripts. It returns true or false depending on the user's rights in
  2953. * this particular course.
  2954. * Optionally checking for tutor and coach roles here allows us to use the
  2955. * student_view feature altogether with these roles as well.
  2956. *
  2957. * @param bool Whether to check if the user has the tutor role
  2958. * @param bool Whether to check if the user has the coach role
  2959. * @param bool Whether to check if the user has the session coach role
  2960. * @param bool check the student view or not
  2961. *
  2962. * @author Roan Embrechts
  2963. * @author Patrick Cool
  2964. * @author Julio Montoya
  2965. *
  2966. * @version 1.1, February 2004
  2967. *
  2968. * @return bool true: the user has the rights to edit, false: he does not
  2969. */
  2970. function api_is_allowed_to_edit(
  2971. $tutor = false,
  2972. $coach = false,
  2973. $session_coach = false,
  2974. $check_student_view = true
  2975. ) {
  2976. $allowSessionAdminEdit = api_get_configuration_value('session_admins_edit_courses_content') === true;
  2977. // Admins can edit anything.
  2978. if (api_is_platform_admin($allowSessionAdminEdit)) {
  2979. //The student preview was on
  2980. if ($check_student_view && api_is_student_view_active()) {
  2981. return false;
  2982. } else {
  2983. return true;
  2984. }
  2985. }
  2986. $sessionId = api_get_session_id();
  2987. if ($sessionId && api_get_configuration_value('session_courses_read_only_mode')) {
  2988. $efv = new ExtraFieldValue('course');
  2989. $lockExrafieldField = $efv->get_values_by_handler_and_field_variable(
  2990. api_get_course_int_id(),
  2991. 'session_courses_read_only_mode'
  2992. );
  2993. if (!empty($lockExrafieldField['value'])) {
  2994. return false;
  2995. }
  2996. }
  2997. $is_allowed_coach_to_edit = api_is_coach(null, null, $check_student_view);
  2998. $session_visibility = api_get_session_visibility($sessionId);
  2999. $is_courseAdmin = api_is_course_admin();
  3000. if (!$is_courseAdmin && $tutor) {
  3001. // If we also want to check if the user is a tutor...
  3002. $is_courseAdmin = $is_courseAdmin || api_is_course_tutor();
  3003. }
  3004. if (!$is_courseAdmin && $coach) {
  3005. // If we also want to check if the user is a coach...';
  3006. // Check if session visibility is read only for coaches.
  3007. if ($session_visibility == SESSION_VISIBLE_READ_ONLY) {
  3008. $is_allowed_coach_to_edit = false;
  3009. }
  3010. if (api_get_setting('allow_coach_to_edit_course_session') == 'true') {
  3011. // Check if coach is allowed to edit a course.
  3012. $is_courseAdmin = $is_courseAdmin || $is_allowed_coach_to_edit;
  3013. }
  3014. }
  3015. if (!$is_courseAdmin && $session_coach) {
  3016. $is_courseAdmin = $is_courseAdmin || $is_allowed_coach_to_edit;
  3017. }
  3018. // Check if the student_view is enabled, and if so, if it is activated.
  3019. if (api_get_setting('student_view_enabled') == 'true') {
  3020. if (!empty($sessionId)) {
  3021. // Check if session visibility is read only for coaches.
  3022. if ($session_visibility == SESSION_VISIBLE_READ_ONLY) {
  3023. $is_allowed_coach_to_edit = false;
  3024. }
  3025. if (api_get_setting('allow_coach_to_edit_course_session') == 'true') {
  3026. // Check if coach is allowed to edit a course.
  3027. $is_allowed = $is_allowed_coach_to_edit;
  3028. } else {
  3029. $is_allowed = false;
  3030. }
  3031. if ($check_student_view) {
  3032. $is_allowed = $is_allowed && $_SESSION['studentview'] != 'studentview';
  3033. }
  3034. } else {
  3035. if ($check_student_view) {
  3036. $is_allowed = $is_courseAdmin && $_SESSION['studentview'] != 'studentview';
  3037. } else {
  3038. $is_allowed = $is_courseAdmin;
  3039. }
  3040. }
  3041. return $is_allowed;
  3042. } else {
  3043. return $is_courseAdmin;
  3044. }
  3045. }
  3046. /**
  3047. * Returns true if user is a course coach of at least one course in session.
  3048. *
  3049. * @param int $sessionId
  3050. *
  3051. * @return bool
  3052. */
  3053. function api_is_coach_of_course_in_session($sessionId)
  3054. {
  3055. if (api_is_platform_admin()) {
  3056. return true;
  3057. }
  3058. $userId = api_get_user_id();
  3059. $courseList = UserManager::get_courses_list_by_session(
  3060. $userId,
  3061. $sessionId
  3062. );
  3063. // Session visibility.
  3064. $visibility = api_get_session_visibility(
  3065. $sessionId,
  3066. null,
  3067. false
  3068. );
  3069. if ($visibility != SESSION_VISIBLE && !empty($courseList)) {
  3070. // Course Coach session visibility.
  3071. $blockedCourseCount = 0;
  3072. $closedVisibilityList = [
  3073. COURSE_VISIBILITY_CLOSED,
  3074. COURSE_VISIBILITY_HIDDEN,
  3075. ];
  3076. foreach ($courseList as $course) {
  3077. // Checking session visibility
  3078. $sessionCourseVisibility = api_get_session_visibility(
  3079. $sessionId,
  3080. $course['real_id']
  3081. );
  3082. $courseIsVisible = !in_array(
  3083. $course['visibility'],
  3084. $closedVisibilityList
  3085. );
  3086. if ($courseIsVisible === false || $sessionCourseVisibility == SESSION_INVISIBLE) {
  3087. $blockedCourseCount++;
  3088. }
  3089. }
  3090. // If all courses are blocked then no show in the list.
  3091. if ($blockedCourseCount === count($courseList)) {
  3092. $visibility = SESSION_INVISIBLE;
  3093. } else {
  3094. $visibility = SESSION_VISIBLE;
  3095. }
  3096. }
  3097. switch ($visibility) {
  3098. case SESSION_VISIBLE_READ_ONLY:
  3099. case SESSION_VISIBLE:
  3100. case SESSION_AVAILABLE:
  3101. return true;
  3102. break;
  3103. case SESSION_INVISIBLE:
  3104. return false;
  3105. }
  3106. return false;
  3107. }
  3108. /**
  3109. * Checks if a student can edit contents in a session depending
  3110. * on the session visibility.
  3111. *
  3112. * @param bool $tutor Whether to check if the user has the tutor role
  3113. * @param bool $coach Whether to check if the user has the coach role
  3114. *
  3115. * @return bool true: the user has the rights to edit, false: he does not
  3116. */
  3117. function api_is_allowed_to_session_edit($tutor = false, $coach = false)
  3118. {
  3119. if (api_is_allowed_to_edit($tutor, $coach)) {
  3120. // If I'm a teacher, I will return true in order to not affect the normal behaviour of Chamilo tools.
  3121. return true;
  3122. } else {
  3123. $sessionId = api_get_session_id();
  3124. if ($sessionId == 0) {
  3125. // I'm not in a session so i will return true to not affect the normal behaviour of Chamilo tools.
  3126. return true;
  3127. } else {
  3128. // I'm in a session and I'm a student
  3129. // Get the session visibility
  3130. $session_visibility = api_get_session_visibility($sessionId);
  3131. // if 5 the session is still available
  3132. switch ($session_visibility) {
  3133. case SESSION_VISIBLE_READ_ONLY: // 1
  3134. return false;
  3135. case SESSION_VISIBLE: // 2
  3136. return true;
  3137. case SESSION_INVISIBLE: // 3
  3138. return false;
  3139. case SESSION_AVAILABLE: //5
  3140. return true;
  3141. }
  3142. }
  3143. }
  3144. }
  3145. /**
  3146. * Checks whether the user is allowed in a specific tool for a specific action.
  3147. *
  3148. * @param string $tool the tool we are checking if the user has a certain permission
  3149. * @param string $action the action we are checking (add, edit, delete, move, visibility)
  3150. *
  3151. * @return bool
  3152. *
  3153. * @author Patrick Cool <patrick.cool@UGent.be>, Ghent University
  3154. * @author Julio Montoya
  3155. *
  3156. * @version 1.0
  3157. */
  3158. function api_is_allowed($tool, $action, $task_id = 0)
  3159. {
  3160. $_user = api_get_user_info();
  3161. $_course = api_get_course_info();
  3162. if (api_is_course_admin()) {
  3163. return true;
  3164. }
  3165. if (is_array($_course) and count($_course) > 0) {
  3166. require_once api_get_path(SYS_CODE_PATH).'permissions/permissions_functions.inc.php';
  3167. // Getting the permissions of this user.
  3168. if ($task_id == 0) {
  3169. $user_permissions = get_permissions('user', $_user['user_id']);
  3170. $_SESSION['total_permissions'][$_course['code']] = $user_permissions;
  3171. }
  3172. // Getting the permissions of the task.
  3173. if ($task_id != 0) {
  3174. $task_permissions = get_permissions('task', $task_id);
  3175. /* !!! */$_SESSION['total_permissions'][$_course['code']] = $task_permissions;
  3176. }
  3177. //print_r($_SESSION['total_permissions']);
  3178. // Getting the permissions of the groups of the user
  3179. //$groups_of_user = GroupManager::get_group_ids($_course['db_name'], $_user['user_id']);
  3180. //foreach($groups_of_user as $group)
  3181. // $this_group_permissions = get_permissions('group', $group);
  3182. // Getting the permissions of the courseroles of the user
  3183. $user_courserole_permissions = get_roles_permissions('user', $_user['user_id']);
  3184. // Getting the permissions of the platformroles of the user
  3185. //$user_platformrole_permissions = get_roles_permissions('user', $_user['user_id'], ', platform');
  3186. // Getting the permissions of the roles of the groups of the user
  3187. //foreach($groups_of_user as $group)
  3188. // $this_group_courserole_permissions = get_roles_permissions('group', $group);
  3189. // Getting the permissions of the platformroles of the groups of the user
  3190. //foreach($groups_of_user as $group)
  3191. // $this_group_platformrole_permissions = get_roles_permissions('group', $group, 'platform');
  3192. }
  3193. // If the permissions are limited, we have to map the extended ones to the limited ones.
  3194. if (api_get_setting('permissions') == 'limited') {
  3195. if ($action == 'Visibility') {
  3196. $action = 'Edit';
  3197. }
  3198. if ($action == 'Move') {
  3199. $action = 'Edit';
  3200. }
  3201. }
  3202. // The session that contains all the permissions already exists for this course
  3203. // so there is no need to requery everything.
  3204. //my_print_r($_SESSION['total_permissions'][$_course['code']][$tool]);
  3205. if (is_array($_SESSION['total_permissions'][$_course['code']][$tool])) {
  3206. if (in_array($action, $_SESSION['total_permissions'][$_course['code']][$tool])) {
  3207. return true;
  3208. } else {
  3209. return false;
  3210. }
  3211. }
  3212. }
  3213. /**
  3214. * Tells whether this user is an anonymous user.
  3215. *
  3216. * @param int $user_id User ID (optional, will take session ID if not provided)
  3217. * @param bool $db_check Whether to check in the database (true) or simply in
  3218. * the session (false) to see if the current user is the anonymous user
  3219. *
  3220. * @return bool true if this user is anonymous, false otherwise
  3221. */
  3222. function api_is_anonymous($user_id = null, $db_check = false)
  3223. {
  3224. if (!isset($user_id)) {
  3225. $user_id = api_get_user_id();
  3226. }
  3227. if ($db_check) {
  3228. $info = api_get_user_info($user_id);
  3229. if ($info['status'] == ANONYMOUS) {
  3230. return true;
  3231. }
  3232. }
  3233. $_user = api_get_user_info();
  3234. if (isset($_user['status']) && $_user['status'] == ANONYMOUS) {
  3235. //if ($_user['user_id'] == 0) {
  3236. // In some cases, api_set_anonymous doesn't seem to be triggered in local.inc.php. Make sure it is.
  3237. // Occurs in agenda for admin links - YW
  3238. global $use_anonymous;
  3239. if (isset($use_anonymous) && $use_anonymous) {
  3240. api_set_anonymous();
  3241. }
  3242. return true;
  3243. }
  3244. return (isset($_user['is_anonymous']) && $_user['is_anonymous'] === true) || $_user === false;
  3245. }
  3246. /**
  3247. * Displays message "You are not allowed here..." and exits the entire script.
  3248. *
  3249. * @param bool $print_headers Whether or not to print headers (default = false -> does not print them)
  3250. * @param string $message
  3251. * @param int $responseCode
  3252. */
  3253. function api_not_allowed(
  3254. $print_headers = false,
  3255. $message = null,
  3256. $responseCode = 0
  3257. ) {
  3258. if (api_get_setting('sso_authentication') === 'true') {
  3259. global $osso;
  3260. if ($osso) {
  3261. $osso->logout();
  3262. }
  3263. }
  3264. $home_url = api_get_path(WEB_PATH);
  3265. $user_id = api_get_user_id();
  3266. $course = api_get_course_id();
  3267. global $this_section;
  3268. if (CustomPages::enabled() && !isset($user_id)) {
  3269. if (empty($user_id)) {
  3270. // Why the CustomPages::enabled() need to be to set the request_uri
  3271. $_SESSION['request_uri'] = $_SERVER['REQUEST_URI'];
  3272. }
  3273. CustomPages::display(CustomPages::INDEX_UNLOGGED);
  3274. }
  3275. $origin = api_get_origin();
  3276. $msg = null;
  3277. if (isset($message)) {
  3278. $msg = $message;
  3279. } else {
  3280. $msg = Display::return_message(
  3281. get_lang('NotAllowedClickBack').'
  3282. <script>function goBack(){window.history.back();}</script>',
  3283. 'error',
  3284. false
  3285. );
  3286. $msg .= '<p class="text-center">
  3287. <a onclick="goBack();" class="btn btn-default" href="'.$home_url.'">'.get_lang('GoBack').'</a>
  3288. </p>';
  3289. }
  3290. $msg = Display::div($msg, ['align' => 'center']);
  3291. $show_headers = 0;
  3292. if ($print_headers && $origin != 'learnpath') {
  3293. $show_headers = 1;
  3294. }
  3295. $tpl = new Template(null, $show_headers, $show_headers, false, true, false, true, $responseCode);
  3296. $tpl->assign('hide_login_link', 1);
  3297. $tpl->assign('content', $msg);
  3298. if (($user_id != 0 && !api_is_anonymous()) &&
  3299. (!isset($course) || $course == -1) &&
  3300. empty($_GET['cidReq'])
  3301. ) {
  3302. // if the access is not authorized and there is some login information
  3303. // but the cidReq is not found, assume we are missing course data and send the user
  3304. // to the user_portal
  3305. $tpl->display_one_col_template();
  3306. exit;
  3307. }
  3308. if (!empty($_SERVER['REQUEST_URI']) &&
  3309. (
  3310. !empty($_GET['cidReq']) ||
  3311. $this_section == SECTION_MYPROFILE ||
  3312. $this_section == SECTION_PLATFORM_ADMIN
  3313. )
  3314. ) {
  3315. $courseCode = api_get_course_id();
  3316. // Only display form and return to the previous URL if there was a course ID included
  3317. if ($user_id != 0 && !api_is_anonymous()) {
  3318. //if there is a user ID, then the user is not allowed but the session is still there. Say so and exit
  3319. $tpl->assign('content', $msg);
  3320. $tpl->display_one_col_template();
  3321. exit;
  3322. }
  3323. if (!is_null($courseCode)) {
  3324. api_set_firstpage_parameter($courseCode);
  3325. }
  3326. // If the user has no user ID, then his session has expired
  3327. $form = api_get_not_allowed_login_form();
  3328. // see same text in auth/gotocourse.php and main_api.lib.php function api_not_allowed (above)
  3329. $content = Display::return_message(get_lang('NotAllowed'), 'error', false);
  3330. if (!empty($courseCode)) {
  3331. $content .= '<h4>'.get_lang('LoginToGoToThisCourse').'</h4>';
  3332. }
  3333. if (api_is_cas_activated()) {
  3334. $content .= Display::return_message(sprintf(get_lang('YouHaveAnInstitutionalAccount'), api_get_setting("Institution")), '', false);
  3335. $content .= Display::div(
  3336. "<br/><a href='".get_cas_direct_URL(api_get_course_id())."'>".sprintf(get_lang('LoginWithYourAccount'), api_get_setting("Institution"))."</a><br/><br/>",
  3337. ['align' => 'center']
  3338. );
  3339. $content .= Display::return_message(get_lang('YouDontHaveAnInstitutionAccount'));
  3340. $content .= "<p style='text-align:center'><a href='#' onclick='$(this).parent().next().toggle()'>".get_lang('LoginWithExternalAccount')."</a></p>";
  3341. $content .= "<div style='display:none;'>";
  3342. }
  3343. $content .= '<div class="well">';
  3344. $content .= $form->returnForm();
  3345. $content .= '</div>';
  3346. if (api_is_cas_activated()) {
  3347. $content .= "</div>";
  3348. }
  3349. if (!empty($courseCode)) {
  3350. $content .= '<hr/><p style="text-align:center"><a href="'.$home_url.'">'.
  3351. get_lang('ReturnToCourseHomepage').'</a></p>';
  3352. } else {
  3353. $content .= '<hr/><p style="text-align:center"><a href="'.$home_url.'">'.
  3354. get_lang('BackHome').'</a></p>';
  3355. }
  3356. $tpl->setLoginBodyClass();
  3357. $tpl->assign('content', $content);
  3358. $tpl->display_one_col_template();
  3359. exit;
  3360. }
  3361. if ($user_id != 0 && !api_is_anonymous()) {
  3362. $tpl->display_one_col_template();
  3363. exit;
  3364. }
  3365. $msg = null;
  3366. // The session is over and we were not in a course,
  3367. // or we try to get directly to a private course without being logged
  3368. $courseId = api_get_course_int_id();
  3369. if (!empty($courseId)) {
  3370. api_set_firstpage_parameter(api_get_course_id());
  3371. $tpl->setLoginBodyClass();
  3372. // see same text in auth/gotocourse.php and main_api.lib.php function api_not_allowed (bellow)
  3373. $msg = Display::return_message(get_lang('NotAllowed'), 'error', false);
  3374. $msg .= '<h4>'.get_lang('LoginToGoToThisCourse').'</h4>';
  3375. $casEnabled = api_is_cas_activated();
  3376. if ($casEnabled) {
  3377. $msg .= Display::return_message(
  3378. sprintf(get_lang('YouHaveAnInstitutionalAccount'), api_get_setting("Institution")),
  3379. '',
  3380. false
  3381. );
  3382. $msg .= Display::div("<br/><a href='".get_cas_direct_URL(api_get_course_int_id())."'>".getCASLogoHTML()." ".sprintf(get_lang('LoginWithYourAccount'), api_get_setting("Institution"))."</a><br/><br/>", ['align' => 'center']);
  3383. $msg .= Display::return_message(get_lang('YouDontHaveAnInstitutionAccount'));
  3384. $msg .= "<p style='text-align:center'><a href='#' onclick='$(this).parent().next().toggle()'>".get_lang('LoginWithExternalAccount')."</a></p>";
  3385. $msg .= "<div style='display:none;'>";
  3386. }
  3387. $form = api_get_not_allowed_login_form();
  3388. $msg .= '<div class="well">';
  3389. $msg .= $form->returnForm();
  3390. $msg .= '</div>';
  3391. if ($casEnabled) {
  3392. $msg .= "</div>";
  3393. }
  3394. } else {
  3395. // we were not in a course, return to home page
  3396. $msg = Display::return_message(
  3397. get_lang('NotAllowed'),
  3398. 'error',
  3399. false
  3400. );
  3401. $msg .= '<p class="text-center">
  3402. <a class="btn btn-default" href="'.$home_url.'">'.get_lang('BackHome').'</a>
  3403. </p>';
  3404. if (!empty($message)) {
  3405. $msg = $message;
  3406. }
  3407. if (api_is_anonymous()) {
  3408. $form = api_get_not_allowed_login_form();
  3409. $msg .= '<div class="well">';
  3410. $msg .= $form->returnForm();
  3411. $msg .= '</div>';
  3412. }
  3413. }
  3414. $tpl->assign('content', $msg);
  3415. $tpl->display_one_col_template();
  3416. exit;
  3417. }
  3418. /**
  3419. * @return FormValidator
  3420. */
  3421. function api_get_not_allowed_login_form()
  3422. {
  3423. $action = api_get_self().'?'.Security::remove_XSS($_SERVER['QUERY_STRING']);
  3424. $action = str_replace('&amp;', '&', $action);
  3425. Session::write('redirect_after_not_allow_page', $action);
  3426. $action .= '&redirect_after_not_allow_page=1';
  3427. $form = new FormValidator(
  3428. 'formLogin',
  3429. 'post',
  3430. $action,
  3431. null,
  3432. ['class' => 'form-stacked']
  3433. );
  3434. $params = [
  3435. 'placeholder' => get_lang('UserName'),
  3436. 'class' => 'col-md-3',
  3437. ];
  3438. if (api_browser_support('autocapitalize')) {
  3439. $params['autocapitalize'] = 'none';
  3440. }
  3441. $form->addElement(
  3442. 'text',
  3443. 'login',
  3444. null,
  3445. $params
  3446. );
  3447. $form->addElement(
  3448. 'password',
  3449. 'password',
  3450. null,
  3451. ['placeholder' => get_lang('Password'), 'class' => 'col-md-3']
  3452. ); //new
  3453. $form->addButtonNext(get_lang('LoginEnter'), 'submitAuth');
  3454. return $form;
  3455. }
  3456. /**
  3457. * Gets a UNIX timestamp from a database (MySQL) datetime format string.
  3458. *
  3459. * @param $last_post_datetime standard output date in a sql query
  3460. *
  3461. * @return int timestamp
  3462. *
  3463. * @author Toon Van Hoecke <Toon.VanHoecke@UGent.be>
  3464. *
  3465. * @version October 2003
  3466. * @desc convert sql date to unix timestamp
  3467. */
  3468. function convert_sql_date($last_post_datetime)
  3469. {
  3470. list($last_post_date, $last_post_time) = explode(' ', $last_post_datetime);
  3471. list($year, $month, $day) = explode('-', $last_post_date);
  3472. list($hour, $min, $sec) = explode(':', $last_post_time);
  3473. return mktime((int) $hour, (int) $min, (int) $sec, (int) $month, (int) $day, (int) $year);
  3474. }
  3475. /**
  3476. * Gets item visibility from the item_property table.
  3477. *
  3478. * Getting the visibility is done by getting the last updated visibility entry,
  3479. * using the largest session ID found if session 0 and another was found (meaning
  3480. * the only one that is actually from the session, in case there are results from
  3481. * session 0 *AND* session n).
  3482. *
  3483. * @param array $_course Course properties array (result of api_get_course_info())
  3484. * @param string $tool Tool (learnpath, document, etc)
  3485. * @param int $id The item ID in the given tool
  3486. * @param int $session The session ID (optional)
  3487. * @param int $user_id
  3488. * @param string $type
  3489. * @param string $group_id
  3490. *
  3491. * @return int -1 on error, 0 if invisible, 1 if visible
  3492. */
  3493. function api_get_item_visibility(
  3494. $_course,
  3495. $tool,
  3496. $id,
  3497. $session = 0,
  3498. $user_id = null,
  3499. $type = null,
  3500. $group_id = null
  3501. ) {
  3502. if (!is_array($_course) || count($_course) == 0 || empty($tool) || empty($id)) {
  3503. return -1;
  3504. }
  3505. $tool = Database::escape_string($tool);
  3506. $id = (int) $id;
  3507. $session = (int) $session;
  3508. $TABLE_ITEMPROPERTY = Database::get_course_table(TABLE_ITEM_PROPERTY);
  3509. $course_id = (int) $_course['real_id'];
  3510. $userCondition = '';
  3511. if (!empty($user_id)) {
  3512. $user_id = (int) $user_id;
  3513. $userCondition = " AND to_user_id = $user_id ";
  3514. }
  3515. $typeCondition = '';
  3516. if (!empty($type)) {
  3517. $type = Database::escape_string($type);
  3518. $typeCondition = " AND lastedit_type = '$type' ";
  3519. }
  3520. $groupCondition = '';
  3521. if (!empty($group_id)) {
  3522. $group_id = (int) $group_id;
  3523. $groupCondition = " AND to_group_id = '$group_id' ";
  3524. }
  3525. $sql = "SELECT visibility
  3526. FROM $TABLE_ITEMPROPERTY
  3527. WHERE
  3528. c_id = $course_id AND
  3529. tool = '$tool' AND
  3530. ref = $id AND
  3531. (session_id = $session OR session_id = 0 OR session_id IS NULL)
  3532. $userCondition $typeCondition $groupCondition
  3533. ORDER BY session_id DESC, lastedit_date DESC
  3534. LIMIT 1";
  3535. $res = Database::query($sql);
  3536. if ($res === false || Database::num_rows($res) == 0) {
  3537. return -1;
  3538. }
  3539. $row = Database::fetch_array($res);
  3540. return (int) $row['visibility'];
  3541. }
  3542. /**
  3543. * Delete a row in the c_item_property table.
  3544. *
  3545. * @param array $courseInfo
  3546. * @param string $tool
  3547. * @param int $itemId
  3548. * @param int $userId
  3549. * @param int $groupId group.iid
  3550. * @param int $sessionId
  3551. *
  3552. * @return false|null
  3553. */
  3554. function api_item_property_delete(
  3555. $courseInfo,
  3556. $tool,
  3557. $itemId,
  3558. $userId,
  3559. $groupId = 0,
  3560. $sessionId = 0
  3561. ) {
  3562. if (empty($courseInfo)) {
  3563. return false;
  3564. }
  3565. $courseId = (int) $courseInfo['real_id'];
  3566. if (empty($courseId) || empty($tool) || empty($itemId)) {
  3567. return false;
  3568. }
  3569. $table = Database::get_course_table(TABLE_ITEM_PROPERTY);
  3570. $tool = Database::escape_string($tool);
  3571. $itemId = intval($itemId);
  3572. $userId = intval($userId);
  3573. $groupId = intval($groupId);
  3574. $sessionId = intval($sessionId);
  3575. $groupCondition = " AND to_group_id = $groupId ";
  3576. if (empty($groupId)) {
  3577. $groupCondition = " AND (to_group_id is NULL OR to_group_id = 0) ";
  3578. }
  3579. $userCondition = " AND to_user_id = $userId ";
  3580. if (empty($userId)) {
  3581. $userCondition = " AND (to_user_id is NULL OR to_user_id = 0) ";
  3582. }
  3583. $sessionCondition = api_get_session_condition($sessionId, true, false, 'session_id');
  3584. $sql = "DELETE FROM $table
  3585. WHERE
  3586. c_id = $courseId AND
  3587. tool = '$tool' AND
  3588. ref = $itemId
  3589. $sessionCondition
  3590. $userCondition
  3591. $groupCondition
  3592. ";
  3593. Database::query($sql);
  3594. }
  3595. /**
  3596. * Updates or adds item properties to the Item_propetry table
  3597. * Tool and lastedit_type are language independant strings (langvars->get_lang!).
  3598. *
  3599. * @param array $_course array with course properties
  3600. * @param string $tool tool id, linked to 'rubrique' of the course tool_list (Warning: language sensitive !!)
  3601. * @param int $item_id id of the item itself, linked to key of every tool ('id', ...)
  3602. * @param string $last_edit_type add or update action
  3603. * (1) message to be translated (in trad4all) : e.g. DocumentAdded, DocumentUpdated;
  3604. * (2) "delete"
  3605. * (3) "visible"
  3606. * (4) "invisible"
  3607. * @param int $user_id id of the editing/adding user
  3608. * @param array $groupInfo must include group.iid/group.od
  3609. * @param int $to_user_id id of the intended user (always has priority over $to_group_id !), only relevant for $type (1)
  3610. * @param string $start_visible 0000-00-00 00:00:00 format
  3611. * @param string $end_visible 0000-00-00 00:00:00 format
  3612. * @param int $session_id The session ID, if any, otherwise will default to 0
  3613. *
  3614. * @return bool false if update fails
  3615. *
  3616. * @author Toon Van Hoecke <Toon.VanHoecke@UGent.be>, Ghent University
  3617. *
  3618. * @version January 2005
  3619. * @desc update the item_properties table (if entry not exists, insert) of the course
  3620. */
  3621. function api_item_property_update(
  3622. $_course,
  3623. $tool,
  3624. $item_id,
  3625. $last_edit_type,
  3626. $user_id,
  3627. $groupInfo = [],
  3628. $to_user_id = null,
  3629. $start_visible = '',
  3630. $end_visible = '',
  3631. $session_id = 0
  3632. ) {
  3633. if (empty($_course)) {
  3634. return false;
  3635. }
  3636. $course_id = $_course['real_id'];
  3637. if (empty($course_id)) {
  3638. return false;
  3639. }
  3640. $to_group_id = 0;
  3641. if (!empty($groupInfo) && isset($groupInfo['iid'])) {
  3642. $to_group_id = (int) $groupInfo['iid'];
  3643. }
  3644. $em = Database::getManager();
  3645. // Definition of variables.
  3646. $tool = Database::escape_string($tool);
  3647. $item_id = (int) $item_id;
  3648. $lastEditTypeNoFilter = $last_edit_type;
  3649. $last_edit_type = Database::escape_string($last_edit_type);
  3650. $user_id = (int) $user_id;
  3651. $startVisible = "NULL";
  3652. if (!empty($start_visible)) {
  3653. $start_visible = Database::escape_string($start_visible);
  3654. $startVisible = "'$start_visible'";
  3655. }
  3656. $endVisible = "NULL";
  3657. if (!empty($end_visible)) {
  3658. $end_visible = Database::escape_string($end_visible);
  3659. $endVisible = "'$end_visible'";
  3660. }
  3661. $to_filter = '';
  3662. $time = api_get_utc_datetime();
  3663. if (!empty($session_id)) {
  3664. $session_id = (int) $session_id;
  3665. } else {
  3666. $session_id = api_get_session_id();
  3667. }
  3668. // Definition of tables.
  3669. $tableItemProperty = Database::get_course_table(TABLE_ITEM_PROPERTY);
  3670. if ($to_user_id <= 0) {
  3671. $to_user_id = null; // No to_user_id set
  3672. }
  3673. if (!is_null($to_user_id)) {
  3674. // $to_user_id has more priority than $to_group_id
  3675. $to_user_id = (int) $to_user_id;
  3676. $to_field = 'to_user_id';
  3677. $to_value = $to_user_id;
  3678. } else {
  3679. // $to_user_id is not set.
  3680. $to_field = 'to_group_id';
  3681. $to_value = $to_group_id;
  3682. }
  3683. $toValueCondition = empty($to_value) ? 'NULL' : "'$to_value'";
  3684. // Set filters for $to_user_id and $to_group_id, with priority for $to_user_id
  3685. $condition_session = " AND session_id = $session_id ";
  3686. if (empty($session_id)) {
  3687. $condition_session = ' AND (session_id = 0 OR session_id IS NULL) ';
  3688. }
  3689. $filter = " c_id = $course_id AND tool = '$tool' AND ref = $item_id $condition_session ";
  3690. // Check whether $to_user_id and $to_group_id are passed in the function call.
  3691. // If both are not passed (both are null) then it is a message for everybody and $to_group_id should be 0 !
  3692. if (is_null($to_user_id) && is_null($to_group_id)) {
  3693. $to_group_id = 0;
  3694. }
  3695. if (!is_null($to_user_id)) {
  3696. // Set filter to intended user.
  3697. $to_filter = " AND to_user_id = $to_user_id $condition_session";
  3698. } else {
  3699. // Set filter to intended group.
  3700. if (($to_group_id != 0) && $to_group_id == strval(intval($to_group_id))) {
  3701. $to_filter = " AND to_group_id = $to_group_id $condition_session";
  3702. }
  3703. }
  3704. // Adding filter if set.
  3705. $filter .= $to_filter;
  3706. // Update if possible
  3707. $set_type = '';
  3708. switch ($lastEditTypeNoFilter) {
  3709. case 'delete':
  3710. // delete = make item only visible for the platform admin.
  3711. $visibility = '2';
  3712. if (!empty($session_id)) {
  3713. // Check whether session id already exist into item_properties for updating visibility or add it.
  3714. $sql = "SELECT session_id FROM $tableItemProperty
  3715. WHERE
  3716. c_id = $course_id AND
  3717. tool = '$tool' AND
  3718. ref = $item_id AND
  3719. session_id = $session_id";
  3720. $rs = Database::query($sql);
  3721. if (Database::num_rows($rs) > 0) {
  3722. $sql = "UPDATE $tableItemProperty
  3723. SET lastedit_type = '".str_replace('_', '', ucwords($tool))."Deleted',
  3724. lastedit_date = '$time',
  3725. lastedit_user_id = $user_id,
  3726. visibility = $visibility,
  3727. session_id = $session_id $set_type
  3728. WHERE $filter";
  3729. $result = Database::query($sql);
  3730. } else {
  3731. $sql = "INSERT INTO $tableItemProperty (c_id, tool, ref, insert_date, insert_user_id, lastedit_date, lastedit_type, lastedit_user_id, $to_field, visibility, start_visible, end_visible, session_id)
  3732. VALUES ($course_id, '$tool',$item_id, '$time', $user_id, '$time', '$last_edit_type',$user_id, $toValueCondition, $visibility, $startVisible, $endVisible, $session_id)";
  3733. $result = Database::query($sql);
  3734. $id = Database::insert_id();
  3735. if ($id) {
  3736. $sql = "UPDATE $tableItemProperty SET id = iid WHERE iid = $id";
  3737. Database::query($sql);
  3738. }
  3739. }
  3740. } else {
  3741. $sql = "UPDATE $tableItemProperty
  3742. SET
  3743. lastedit_type='".str_replace('_', '', ucwords($tool))."Deleted',
  3744. lastedit_date='$time',
  3745. lastedit_user_id = $user_id,
  3746. visibility = $visibility $set_type
  3747. WHERE $filter";
  3748. $result = Database::query($sql);
  3749. }
  3750. break;
  3751. case 'visible': // Change item to visible.
  3752. $visibility = '1';
  3753. if (!empty($session_id)) {
  3754. // Check whether session id already exist into item_properties for updating visibility or add it.
  3755. $sql = "SELECT session_id FROM $tableItemProperty
  3756. WHERE
  3757. c_id = $course_id AND
  3758. tool = '$tool' AND
  3759. ref = $item_id AND
  3760. session_id = $session_id";
  3761. $rs = Database::query($sql);
  3762. if (Database::num_rows($rs) > 0) {
  3763. $sql = "UPDATE $tableItemProperty
  3764. SET
  3765. lastedit_type='".str_replace('_', '', ucwords($tool))."Visible',
  3766. lastedit_date='$time',
  3767. lastedit_user_id = $user_id,
  3768. visibility = $visibility,
  3769. session_id = $session_id $set_type
  3770. WHERE $filter";
  3771. $result = Database::query($sql);
  3772. } else {
  3773. $sql = "INSERT INTO $tableItemProperty (c_id, tool, ref, insert_date, insert_user_id, lastedit_date, lastedit_type, lastedit_user_id, $to_field, visibility, start_visible, end_visible, session_id)
  3774. VALUES ($course_id, '$tool', $item_id, '$time', $user_id, '$time', '$last_edit_type', $user_id, $toValueCondition, $visibility, $startVisible, $endVisible, $session_id)";
  3775. $result = Database::query($sql);
  3776. $id = Database::insert_id();
  3777. if ($id) {
  3778. $sql = "UPDATE $tableItemProperty SET id = iid WHERE iid = $id";
  3779. Database::query($sql);
  3780. }
  3781. }
  3782. } else {
  3783. $sql = "UPDATE $tableItemProperty
  3784. SET
  3785. lastedit_type='".str_replace('_', '', ucwords($tool))."Visible',
  3786. lastedit_date='$time',
  3787. lastedit_user_id = $user_id,
  3788. visibility = $visibility $set_type
  3789. WHERE $filter";
  3790. $result = Database::query($sql);
  3791. }
  3792. break;
  3793. case 'invisible': // Change item to invisible.
  3794. $visibility = '0';
  3795. if (!empty($session_id)) {
  3796. // Check whether session id already exist into item_properties for updating visibility or add it
  3797. $sql = "SELECT session_id FROM $tableItemProperty
  3798. WHERE
  3799. c_id = $course_id AND
  3800. tool = '$tool' AND
  3801. ref = $item_id AND
  3802. session_id = $session_id";
  3803. $rs = Database::query($sql);
  3804. if (Database::num_rows($rs) > 0) {
  3805. $sql = "UPDATE $tableItemProperty
  3806. SET
  3807. lastedit_type = '".str_replace('_', '', ucwords($tool))."Invisible',
  3808. lastedit_date = '$time',
  3809. lastedit_user_id = $user_id,
  3810. visibility = $visibility,
  3811. session_id = $session_id $set_type
  3812. WHERE $filter";
  3813. $result = Database::query($sql);
  3814. } else {
  3815. $sql = "INSERT INTO $tableItemProperty (c_id, tool, ref, insert_date, insert_user_id, lastedit_date, lastedit_type, lastedit_user_id,$to_field, visibility, start_visible, end_visible, session_id)
  3816. VALUES ($course_id, '$tool', $item_id, '$time', $user_id, '$time', '$last_edit_type', $user_id, $toValueCondition, $visibility, $startVisible, $endVisible, $session_id)";
  3817. $result = Database::query($sql);
  3818. $id = Database::insert_id();
  3819. if ($id) {
  3820. $sql = "UPDATE $tableItemProperty SET id = iid WHERE iid = $id";
  3821. Database::query($sql);
  3822. }
  3823. }
  3824. } else {
  3825. $sql = "UPDATE $tableItemProperty
  3826. SET
  3827. lastedit_type = '".str_replace('_', '', ucwords($tool))."Invisible',
  3828. lastedit_date = '$time',
  3829. lastedit_user_id = $user_id,
  3830. visibility = $visibility $set_type
  3831. WHERE $filter";
  3832. $result = Database::query($sql);
  3833. }
  3834. break;
  3835. default: // The item will be added or updated.
  3836. $set_type = ", lastedit_type = '$last_edit_type' ";
  3837. $visibility = '1';
  3838. //$filter .= $to_filter; already added
  3839. $sql = "UPDATE $tableItemProperty
  3840. SET
  3841. lastedit_date = '$time',
  3842. lastedit_user_id = $user_id $set_type
  3843. WHERE $filter";
  3844. $result = Database::query($sql);
  3845. }
  3846. // Insert if no entries are found (can only happen in case of $last_edit_type switch is 'default').
  3847. if ($result == false || Database::affected_rows($result) == 0) {
  3848. $objCourse = $em->find('ChamiloCoreBundle:Course', intval($course_id));
  3849. $objTime = new DateTime('now', new DateTimeZone('UTC'));
  3850. $objUser = api_get_user_entity($user_id);
  3851. if (empty($objUser)) {
  3852. // Use anonymous
  3853. $user_id = api_get_anonymous_id();
  3854. $objUser = api_get_user_entity($user_id);
  3855. }
  3856. $objGroup = null;
  3857. if (!empty($to_group_id)) {
  3858. $objGroup = $em->find('ChamiloCourseBundle:CGroupInfo', $to_group_id);
  3859. }
  3860. $objToUser = api_get_user_entity($to_user_id);
  3861. $objSession = $em->find('ChamiloCoreBundle:Session', intval($session_id));
  3862. $startVisibleDate = !empty($start_visible) ? new DateTime($start_visible, new DateTimeZone('UTC')) : null;
  3863. $endVisibleDate = !empty($endVisibleDate) ? new DateTime($endVisibleDate, new DateTimeZone('UTC')) : null;
  3864. $cItemProperty = new CItemProperty($objCourse);
  3865. $cItemProperty
  3866. ->setTool($tool)
  3867. ->setRef($item_id)
  3868. ->setInsertDate($objTime)
  3869. ->setInsertUser($objUser)
  3870. ->setLasteditDate($objTime)
  3871. ->setLasteditType($last_edit_type)
  3872. ->setGroup($objGroup)
  3873. ->setToUser($objToUser)
  3874. ->setVisibility($visibility)
  3875. ->setStartVisible($startVisibleDate)
  3876. ->setEndVisible($endVisibleDate)
  3877. ->setSession($objSession);
  3878. $em->persist($cItemProperty);
  3879. $em->flush();
  3880. $id = $cItemProperty->getIid();
  3881. if ($id) {
  3882. $cItemProperty->setId($id);
  3883. $em->merge($cItemProperty);
  3884. $em->flush();
  3885. return false;
  3886. }
  3887. }
  3888. return true;
  3889. }
  3890. /**
  3891. * Gets item property by tool.
  3892. *
  3893. * @param string course code
  3894. * @param string tool name, linked to 'rubrique' of the course tool_list (Warning: language sensitive !!)
  3895. * @param int id of the item itself, linked to key of every tool ('id', ...), "*" = all items of the tool
  3896. * @param int $session_id
  3897. * @param string $tool
  3898. * @param string $course_code
  3899. *
  3900. * @return array All fields from c_item_property (all rows found) or empty array
  3901. */
  3902. function api_get_item_property_by_tool($tool, $course_code, $session_id = null)
  3903. {
  3904. $course_info = api_get_course_info($course_code);
  3905. $tool = Database::escape_string($tool);
  3906. // Definition of tables.
  3907. $item_property_table = Database::get_course_table(TABLE_ITEM_PROPERTY);
  3908. $session_id = (int) $session_id;
  3909. $session_condition = ' AND session_id = '.$session_id;
  3910. if (empty($session_id)) {
  3911. $session_condition = " AND (session_id = 0 OR session_id IS NULL) ";
  3912. }
  3913. $course_id = $course_info['real_id'];
  3914. $sql = "SELECT * FROM $item_property_table
  3915. WHERE
  3916. c_id = $course_id AND
  3917. tool = '$tool'
  3918. $session_condition ";
  3919. $rs = Database::query($sql);
  3920. $list = [];
  3921. if (Database::num_rows($rs) > 0) {
  3922. while ($row = Database::fetch_array($rs, 'ASSOC')) {
  3923. $list[] = $row;
  3924. }
  3925. }
  3926. return $list;
  3927. }
  3928. /**
  3929. * Gets item property by tool and user.
  3930. *
  3931. * @param int $userId
  3932. * @param int $tool
  3933. * @param int $courseId
  3934. * @param int $session_id
  3935. *
  3936. * @return array
  3937. */
  3938. function api_get_item_property_list_by_tool_by_user(
  3939. $userId,
  3940. $tool,
  3941. $courseId,
  3942. $session_id = 0
  3943. ) {
  3944. $userId = intval($userId);
  3945. $tool = Database::escape_string($tool);
  3946. $session_id = intval($session_id);
  3947. $courseId = intval($courseId);
  3948. // Definition of tables.
  3949. $item_property_table = Database::get_course_table(TABLE_ITEM_PROPERTY);
  3950. $session_condition = ' AND session_id = '.$session_id;
  3951. if (empty($session_id)) {
  3952. $session_condition = " AND (session_id = 0 OR session_id IS NULL) ";
  3953. }
  3954. $sql = "SELECT * FROM $item_property_table
  3955. WHERE
  3956. insert_user_id = $userId AND
  3957. c_id = $courseId AND
  3958. tool = '$tool'
  3959. $session_condition ";
  3960. $rs = Database::query($sql);
  3961. $list = [];
  3962. if (Database::num_rows($rs) > 0) {
  3963. while ($row = Database::fetch_array($rs, 'ASSOC')) {
  3964. $list[] = $row;
  3965. }
  3966. }
  3967. return $list;
  3968. }
  3969. /**
  3970. * Gets item property id from tool of a course.
  3971. *
  3972. * @param string $course_code course code
  3973. * @param string $tool tool name, linked to 'rubrique' of the course tool_list (Warning: language sensitive !!)
  3974. * @param int $ref id of the item itself, linked to key of every tool ('id', ...), "*" = all items of the tool
  3975. * @param int $sessionId Session ID (optional)
  3976. *
  3977. * @return int
  3978. */
  3979. function api_get_item_property_id($course_code, $tool, $ref, $sessionId = 0)
  3980. {
  3981. $course_info = api_get_course_info($course_code);
  3982. $tool = Database::escape_string($tool);
  3983. $ref = (int) $ref;
  3984. // Definition of tables.
  3985. $tableItemProperty = Database::get_course_table(TABLE_ITEM_PROPERTY);
  3986. $course_id = $course_info['real_id'];
  3987. $sessionId = (int) $sessionId;
  3988. $sessionCondition = " AND session_id = $sessionId ";
  3989. if (empty($sessionId)) {
  3990. $sessionCondition = ' AND (session_id = 0 OR session_id IS NULL) ';
  3991. }
  3992. $sql = "SELECT id FROM $tableItemProperty
  3993. WHERE
  3994. c_id = $course_id AND
  3995. tool = '$tool' AND
  3996. ref = $ref
  3997. $sessionCondition";
  3998. $rs = Database::query($sql);
  3999. $item_property_id = '';
  4000. if (Database::num_rows($rs) > 0) {
  4001. $row = Database::fetch_array($rs);
  4002. $item_property_id = $row['id'];
  4003. }
  4004. return $item_property_id;
  4005. }
  4006. /**
  4007. * Inserts a record in the track_e_item_property table (No update).
  4008. *
  4009. * @param string $tool
  4010. * @param int $ref
  4011. * @param string $title
  4012. * @param string $content
  4013. * @param int $progress
  4014. *
  4015. * @return bool|int
  4016. */
  4017. function api_track_item_property_update($tool, $ref, $title, $content, $progress)
  4018. {
  4019. $tbl_stats_item_property = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ITEM_PROPERTY);
  4020. $course_id = api_get_course_int_id(); //numeric
  4021. $course_code = api_get_course_id(); //alphanumeric
  4022. $item_property_id = api_get_item_property_id($course_code, $tool, $ref);
  4023. if (!empty($item_property_id)) {
  4024. $sql = "INSERT IGNORE INTO $tbl_stats_item_property SET
  4025. course_id = '$course_id',
  4026. item_property_id = '$item_property_id',
  4027. title = '".Database::escape_string($title)."',
  4028. content = '".Database::escape_string($content)."',
  4029. progress = '".intval($progress)."',
  4030. lastedit_date = '".api_get_utc_datetime()."',
  4031. lastedit_user_id = '".api_get_user_id()."',
  4032. session_id = '".api_get_session_id()."'";
  4033. $result = Database::query($sql);
  4034. $affected_rows = Database::affected_rows($result);
  4035. return $affected_rows;
  4036. }
  4037. return false;
  4038. }
  4039. /**
  4040. * @param string $tool
  4041. * @param int $ref
  4042. *
  4043. * @return array|resource
  4044. */
  4045. function api_get_track_item_property_history($tool, $ref)
  4046. {
  4047. $tbl_stats_item_property = Database::get_main_table(TABLE_STATISTIC_TRACK_E_ITEM_PROPERTY);
  4048. $course_id = api_get_course_int_id(); //numeric
  4049. $course_code = api_get_course_id(); //alphanumeric
  4050. $item_property_id = api_get_item_property_id($course_code, $tool, $ref);
  4051. $sql = "SELECT * FROM $tbl_stats_item_property
  4052. WHERE item_property_id = $item_property_id AND course_id = $course_id
  4053. ORDER BY lastedit_date DESC";
  4054. $result = Database::query($sql);
  4055. if ($result === false or $result === null) {
  4056. $result = [];
  4057. } else {
  4058. $result = Database::store_result($result, 'ASSOC');
  4059. }
  4060. return $result;
  4061. }
  4062. /**
  4063. * Gets item property data from tool of a course id.
  4064. *
  4065. * @param int $course_id
  4066. * @param string $tool tool name, linked to 'rubrique' of the course tool_list (Warning: language sensitive !!)
  4067. * @param int $ref id of the item itself, linked to key of every tool ('id', ...), "*" = all items of the tool
  4068. * @param int $session_id
  4069. * @param int $groupId
  4070. *
  4071. * @return array with all fields from c_item_property, empty array if not found or false if course could not be found
  4072. */
  4073. function api_get_item_property_info($course_id, $tool, $ref, $session_id = 0, $groupId = 0)
  4074. {
  4075. $courseInfo = api_get_course_info_by_id($course_id);
  4076. if (empty($courseInfo)) {
  4077. return false;
  4078. }
  4079. $tool = Database::escape_string($tool);
  4080. $course_id = $courseInfo['real_id'];
  4081. $ref = (int) $ref;
  4082. $session_id = (int) $session_id;
  4083. $sessionCondition = " session_id = $session_id";
  4084. if (empty($session_id)) {
  4085. $sessionCondition = ' (session_id = 0 OR session_id IS NULL) ';
  4086. }
  4087. // Definition of tables.
  4088. $table = Database::get_course_table(TABLE_ITEM_PROPERTY);
  4089. $sql = "SELECT * FROM $table
  4090. WHERE
  4091. c_id = $course_id AND
  4092. tool = '$tool' AND
  4093. ref = $ref AND
  4094. $sessionCondition ";
  4095. if (!empty($groupId)) {
  4096. $groupId = (int) $groupId;
  4097. $sql .= " AND to_group_id = $groupId ";
  4098. }
  4099. $rs = Database::query($sql);
  4100. $row = [];
  4101. if (Database::num_rows($rs) > 0) {
  4102. $row = Database::fetch_array($rs, 'ASSOC');
  4103. }
  4104. return $row;
  4105. }
  4106. /**
  4107. * Displays a combo box so the user can select his/her preferred language.
  4108. *
  4109. * @param string The desired name= value for the select
  4110. * @param bool Whether we use the JQuery Chozen library or not
  4111. * (in some cases, like the indexing language picker, it can alter the presentation)
  4112. *
  4113. * @return string
  4114. */
  4115. function api_get_languages_combo($name = 'language')
  4116. {
  4117. $ret = '';
  4118. $platformLanguage = api_get_setting('platformLanguage');
  4119. // Retrieve a complete list of all the languages.
  4120. $language_list = api_get_languages();
  4121. if (count($language_list['name']) < 2) {
  4122. return $ret;
  4123. }
  4124. // The the current language of the user so that his/her language occurs as selected in the dropdown menu.
  4125. if (isset($_SESSION['user_language_choice'])) {
  4126. $default = $_SESSION['user_language_choice'];
  4127. } else {
  4128. $default = $platformLanguage;
  4129. }
  4130. $languages = $language_list['name'];
  4131. $folder = $language_list['folder'];
  4132. $ret .= '<select name="'.$name.'" id="language_chosen" class="selectpicker show-tick form-control">';
  4133. foreach ($languages as $key => $value) {
  4134. if ($folder[$key] == $default) {
  4135. $selected = ' selected="selected"';
  4136. } else {
  4137. $selected = '';
  4138. }
  4139. $ret .= sprintf('<option value=%s" %s>%s</option>', $folder[$key], $selected, $value);
  4140. }
  4141. $ret .= '</select>';
  4142. return $ret;
  4143. }
  4144. /**
  4145. * Displays a form (drop down menu) so the user can select his/her preferred language.
  4146. * The form works with or without javascript.
  4147. *
  4148. * @param bool Hide form if only one language available (defaults to false = show the box anyway)
  4149. * @param bool $showAsButton
  4150. *
  4151. * @return string|null Display the box directly
  4152. */
  4153. function api_display_language_form($hide_if_no_choice = false, $showAsButton = false)
  4154. {
  4155. // Retrieve a complete list of all the languages.
  4156. $language_list = api_get_languages();
  4157. if (count($language_list['name']) <= 1 && $hide_if_no_choice) {
  4158. return; //don't show any form
  4159. }
  4160. // The the current language of the user so that his/her language occurs as selected in the dropdown menu.
  4161. if (isset($_SESSION['user_language_choice'])) {
  4162. $user_selected_language = $_SESSION['user_language_choice'];
  4163. }
  4164. if (empty($user_selected_language)) {
  4165. $user_selected_language = api_get_setting('platformLanguage');
  4166. }
  4167. $currentLanguageId = api_get_language_id($user_selected_language);
  4168. $currentLanguageInfo = api_get_language_info($currentLanguageId);
  4169. $countryCode = languageToCountryIsoCode($currentLanguageInfo['isocode']);
  4170. $url = api_get_self();
  4171. if ($showAsButton) {
  4172. $html = '<div class="btn-group">
  4173. <button type="button" class="btn btn-default dropdown-toggle" data-toggle="dropdown">
  4174. <span class="flag-icon flag-icon-'.$countryCode.'"></span>
  4175. '.$currentLanguageInfo['original_name'].'
  4176. <span class="caret">
  4177. </span>
  4178. </button>';
  4179. } else {
  4180. $html = '
  4181. <a href="'.$url.'" class="dropdown-toggle" data-toggle="dropdown" role="button">
  4182. <span class="flag-icon flag-icon-'.$countryCode.'"></span>
  4183. '.$currentLanguageInfo['original_name'].'
  4184. <span class="caret"></span>
  4185. </a>
  4186. ';
  4187. }
  4188. $html .= '<ul class="dropdown-menu" role="menu">';
  4189. foreach ($language_list['all'] as $key => $data) {
  4190. $urlLink = $url.'?language='.$data['english_name'];
  4191. $html .= '<li><a href="'.$urlLink.'"><span class="flag-icon flag-icon-'.languageToCountryIsoCode($data['isocode']).'"></span> '.$data['original_name'].'</a></li>';
  4192. }
  4193. $html .= '</ul>';
  4194. if ($showAsButton) {
  4195. $html .= '</div>';
  4196. }
  4197. return $html;
  4198. }
  4199. /**
  4200. * @param string $languageIsoCode
  4201. *
  4202. * @return string
  4203. */
  4204. function languageToCountryIsoCode($languageIsoCode)
  4205. {
  4206. $allow = api_get_configuration_value('language_flags_by_country');
  4207. // @todo save in DB
  4208. switch ($languageIsoCode) {
  4209. case 'ko':
  4210. $country = 'kr';
  4211. break;
  4212. case 'ja':
  4213. $country = 'jp';
  4214. break;
  4215. case 'ca':
  4216. $country = 'es';
  4217. if ($allow) {
  4218. $country = 'catalan';
  4219. }
  4220. break;
  4221. case 'gl': // galego
  4222. $country = 'es';
  4223. if ($allow) {
  4224. $country = 'galician';
  4225. }
  4226. break;
  4227. case 'ka':
  4228. $country = 'ge';
  4229. break;
  4230. case 'sl':
  4231. $country = 'si';
  4232. break;
  4233. case 'eu': // Euskera
  4234. $country = 'es';
  4235. if ($allow) {
  4236. $country = 'basque';
  4237. }
  4238. break;
  4239. case 'cs':
  4240. $country = 'cz';
  4241. break;
  4242. case 'el':
  4243. $country = 'ae';
  4244. break;
  4245. case 'ar':
  4246. $country = 'ae';
  4247. break;
  4248. case 'en':
  4249. $country = 'gb';
  4250. break;
  4251. case 'he':
  4252. $country = 'il';
  4253. break;
  4254. case 'uk': // Ukraine
  4255. $country = 'ua';
  4256. break;
  4257. case 'da':
  4258. $country = 'dk';
  4259. break;
  4260. case 'pt-BR':
  4261. $country = 'br';
  4262. break;
  4263. case 'qu':
  4264. $country = 'pe';
  4265. break;
  4266. case 'sv':
  4267. $country = 'se';
  4268. break;
  4269. case 'zh-TW':
  4270. case 'zh':
  4271. $country = 'cn';
  4272. break;
  4273. default:
  4274. $country = $languageIsoCode;
  4275. break;
  4276. }
  4277. $country = strtolower($country);
  4278. return $country;
  4279. }
  4280. /**
  4281. * Returns a list of all the languages that are made available by the admin.
  4282. *
  4283. * @return array An array with all languages. Structure of the array is
  4284. * array['name'] = An array with the name of every language
  4285. * array['folder'] = An array with the corresponding names of the language-folders in the filesystem
  4286. */
  4287. function api_get_languages()
  4288. {
  4289. $tbl_language = Database::get_main_table(TABLE_MAIN_LANGUAGE);
  4290. $sql = "SELECT * FROM $tbl_language WHERE available='1'
  4291. ORDER BY original_name ASC";
  4292. $result = Database::query($sql);
  4293. $language_list = [];
  4294. while ($row = Database::fetch_array($result)) {
  4295. $language_list['name'][] = $row['original_name'];
  4296. $language_list['folder'][] = $row['dokeos_folder'];
  4297. $language_list['all'][] = $row;
  4298. }
  4299. return $language_list;
  4300. }
  4301. /**
  4302. * Returns a list of all the languages that are made available by the admin.
  4303. *
  4304. * @return array
  4305. */
  4306. function api_get_languages_to_array()
  4307. {
  4308. $tbl_language = Database::get_main_table(TABLE_MAIN_LANGUAGE);
  4309. $sql = "SELECT * FROM $tbl_language WHERE available='1' ORDER BY original_name ASC";
  4310. $result = Database::query($sql);
  4311. $languages = [];
  4312. while ($row = Database::fetch_array($result)) {
  4313. $languages[$row['dokeos_folder']] = $row['original_name'];
  4314. }
  4315. return $languages;
  4316. }
  4317. /**
  4318. * Returns the id (the database id) of a language.
  4319. *
  4320. * @param string language name (the corresponding name of the language-folder in the filesystem)
  4321. *
  4322. * @return int id of the language
  4323. */
  4324. function api_get_language_id($language)
  4325. {
  4326. $tbl_language = Database::get_main_table(TABLE_MAIN_LANGUAGE);
  4327. if (empty($language)) {
  4328. return null;
  4329. }
  4330. $language = Database::escape_string($language);
  4331. $sql = "SELECT id FROM $tbl_language
  4332. WHERE dokeos_folder = '$language' LIMIT 1";
  4333. $result = Database::query($sql);
  4334. $row = Database::fetch_array($result);
  4335. return $row['id'];
  4336. }
  4337. /**
  4338. * Gets language of the requested type for the current user. Types are :
  4339. * user_profil_lang : profile language of current user
  4340. * user_select_lang : language selected by user at login
  4341. * course_lang : language of the current course
  4342. * platform_lang : default platform language.
  4343. *
  4344. * @param string $lang_type
  4345. *
  4346. * @return string
  4347. */
  4348. function api_get_language_from_type($lang_type)
  4349. {
  4350. $return = false;
  4351. switch ($lang_type) {
  4352. case 'platform_lang':
  4353. $temp_lang = api_get_setting('platformLanguage');
  4354. if (!empty($temp_lang)) {
  4355. $return = $temp_lang;
  4356. }
  4357. break;
  4358. case 'user_profil_lang':
  4359. $_user = api_get_user_info();
  4360. if (isset($_user['language']) && !empty($_user['language'])) {
  4361. $return = $_user['language'];
  4362. }
  4363. break;
  4364. case 'user_selected_lang':
  4365. if (isset($_SESSION['user_language_choice']) && !empty($_SESSION['user_language_choice'])) {
  4366. $return = $_SESSION['user_language_choice'];
  4367. }
  4368. break;
  4369. case 'course_lang':
  4370. global $_course;
  4371. $cidReq = null;
  4372. if (empty($_course)) {
  4373. // Code modified because the local.inc.php file it's declarated after this work
  4374. // causing the function api_get_course_info() returns a null value
  4375. $cidReq = isset($_GET["cidReq"]) ? Database::escape_string($_GET["cidReq"]) : null;
  4376. $cDir = (!empty($_GET['cDir']) ? $_GET['cDir'] : null);
  4377. if (empty($cidReq) && !empty($cDir)) {
  4378. $c = CourseManager::getCourseCodeFromDirectory($cDir);
  4379. if ($c) {
  4380. $cidReq = $c;
  4381. }
  4382. }
  4383. }
  4384. $_course = api_get_course_info($cidReq);
  4385. if (isset($_course['language']) && !empty($_course['language'])) {
  4386. $return = $_course['language'];
  4387. $showCourseInUserLanguage = api_get_course_setting('show_course_in_user_language');
  4388. if ($showCourseInUserLanguage == 1) {
  4389. $userInfo = api_get_user_info();
  4390. if (isset($userInfo['language'])) {
  4391. $return = $userInfo['language'];
  4392. }
  4393. }
  4394. }
  4395. break;
  4396. default:
  4397. $return = false;
  4398. break;
  4399. }
  4400. return $return;
  4401. }
  4402. /**
  4403. * Get the language information by its id.
  4404. *
  4405. * @param int $languageId
  4406. *
  4407. * @throws Exception
  4408. *
  4409. * @return array
  4410. */
  4411. function api_get_language_info($languageId)
  4412. {
  4413. if (empty($languageId)) {
  4414. return [];
  4415. }
  4416. $language = Database::getManager()
  4417. ->find('ChamiloCoreBundle:Language', $languageId);
  4418. if (!$language) {
  4419. return [];
  4420. }
  4421. return [
  4422. 'id' => $language->getId(),
  4423. 'original_name' => $language->getOriginalName(),
  4424. 'english_name' => $language->getEnglishName(),
  4425. 'isocode' => $language->getIsocode(),
  4426. 'dokeos_folder' => $language->getDokeosFolder(),
  4427. 'available' => $language->getAvailable(),
  4428. 'parent_id' => $language->getParent() ? $language->getParent()->getId() : null,
  4429. ];
  4430. }
  4431. /**
  4432. * Returns the name of the visual (CSS) theme to be applied on the current page.
  4433. * The returned name depends on the platform, course or user -wide settings.
  4434. *
  4435. * @return string The visual theme's name, it is the name of a folder inside web/css/themes
  4436. */
  4437. function api_get_visual_theme()
  4438. {
  4439. static $visual_theme;
  4440. if (!isset($visual_theme)) {
  4441. // Get style directly from DB
  4442. $styleFromDatabase = api_get_settings_params_simple(
  4443. [
  4444. 'variable = ? AND access_url = ?' => [
  4445. 'stylesheets',
  4446. api_get_current_access_url_id(),
  4447. ],
  4448. ]
  4449. );
  4450. if ($styleFromDatabase) {
  4451. $platform_theme = $styleFromDatabase['selected_value'];
  4452. } else {
  4453. $platform_theme = api_get_setting('stylesheets');
  4454. }
  4455. // Platform's theme.
  4456. $visual_theme = $platform_theme;
  4457. if (api_get_setting('user_selected_theme') == 'true') {
  4458. $user_info = api_get_user_info();
  4459. if (isset($user_info['theme'])) {
  4460. $user_theme = $user_info['theme'];
  4461. if (!empty($user_theme)) {
  4462. $visual_theme = $user_theme;
  4463. // User's theme.
  4464. }
  4465. }
  4466. }
  4467. $course_id = api_get_course_id();
  4468. if (!empty($course_id)) {
  4469. if (api_get_setting('allow_course_theme') == 'true') {
  4470. $course_theme = api_get_course_setting('course_theme', $course_id);
  4471. if (!empty($course_theme) && $course_theme != -1) {
  4472. if (!empty($course_theme)) {
  4473. // Course's theme.
  4474. $visual_theme = $course_theme;
  4475. }
  4476. }
  4477. $allow_lp_theme = api_get_course_setting('allow_learning_path_theme');
  4478. if ($allow_lp_theme == 1) {
  4479. global $lp_theme_css, $lp_theme_config;
  4480. // These variables come from the file lp_controller.php.
  4481. if (!$lp_theme_config) {
  4482. if (!empty($lp_theme_css)) {
  4483. // LP's theme.
  4484. $visual_theme = $lp_theme_css;
  4485. }
  4486. }
  4487. }
  4488. }
  4489. }
  4490. if (empty($visual_theme)) {
  4491. $visual_theme = 'chamilo';
  4492. }
  4493. global $lp_theme_log;
  4494. if ($lp_theme_log) {
  4495. $visual_theme = $platform_theme;
  4496. }
  4497. }
  4498. return $visual_theme;
  4499. }
  4500. /**
  4501. * Returns a list of CSS themes currently available in the CSS folder
  4502. * The folder must have a default.css file.
  4503. *
  4504. * @param bool $getOnlyThemeFromVirtualInstance Used by the vchamilo plugin
  4505. *
  4506. * @return array list of themes directories from the css folder
  4507. * Note: Directory names (names of themes) in the file system should contain ASCII-characters only
  4508. */
  4509. function api_get_themes($getOnlyThemeFromVirtualInstance = false)
  4510. {
  4511. // This configuration value is set by the vchamilo plugin
  4512. $virtualTheme = api_get_configuration_value('virtual_css_theme_folder');
  4513. $readCssFolder = function ($dir) use ($virtualTheme) {
  4514. $finder = new Finder();
  4515. $themes = $finder->directories()->in($dir)->depth(0)->sortByName();
  4516. $list = [];
  4517. /** @var Symfony\Component\Finder\SplFileInfo $theme */
  4518. foreach ($themes as $theme) {
  4519. $folder = $theme->getFilename();
  4520. // A theme folder is consider if there's a default.css file
  4521. if (!file_exists($theme->getPathname().'/default.css')) {
  4522. continue;
  4523. }
  4524. $name = ucwords(str_replace('_', ' ', $folder));
  4525. if ($folder == $virtualTheme) {
  4526. continue;
  4527. }
  4528. $list[$folder] = $name;
  4529. }
  4530. return $list;
  4531. };
  4532. $dir = api_get_path(SYS_CSS_PATH).'themes/';
  4533. $list = $readCssFolder($dir);
  4534. if (!empty($virtualTheme)) {
  4535. $newList = $readCssFolder($dir.'/'.$virtualTheme);
  4536. if ($getOnlyThemeFromVirtualInstance) {
  4537. return $newList;
  4538. }
  4539. $list = $list + $newList;
  4540. asort($list);
  4541. }
  4542. return $list;
  4543. }
  4544. /**
  4545. * Find the largest sort value in a given user_course_category
  4546. * This function is used when we are moving a course to a different category
  4547. * and also when a user subscribes to courses (the new course is added at the end of the main category.
  4548. *
  4549. * @author Patrick Cool <patrick.cool@UGent.be>, Ghent University
  4550. *
  4551. * @param int $user_course_category the id of the user_course_category
  4552. * @param int $user_id
  4553. *
  4554. * @return int the value of the highest sort of the user_course_category
  4555. */
  4556. function api_max_sort_value($user_course_category, $user_id)
  4557. {
  4558. $tbl_course_user = Database::get_main_table(TABLE_MAIN_COURSE_USER);
  4559. $sql = "SELECT max(sort) as max_sort FROM $tbl_course_user
  4560. WHERE
  4561. user_id='".intval($user_id)."' AND
  4562. relation_type<>".COURSE_RELATION_TYPE_RRHH." AND
  4563. user_course_cat='".intval($user_course_category)."'";
  4564. $result_max = Database::query($sql);
  4565. if (Database::num_rows($result_max) == 1) {
  4566. $row_max = Database::fetch_array($result_max);
  4567. return $row_max['max_sort'];
  4568. }
  4569. return 0;
  4570. }
  4571. /**
  4572. * Transforms a number of seconds in hh:mm:ss format.
  4573. *
  4574. * @author Julian Prud'homme
  4575. *
  4576. * @param int the number of seconds
  4577. *
  4578. * @return string the formated time
  4579. */
  4580. function api_time_to_hms($seconds, $space = ':')
  4581. {
  4582. // $seconds = -1 means that we have wrong data in the db.
  4583. if ($seconds == -1) {
  4584. return
  4585. get_lang('Unknown').
  4586. Display::return_icon(
  4587. 'info2.gif',
  4588. get_lang('WrongDatasForTimeSpentOnThePlatform'),
  4589. ['align' => 'absmiddle', 'hspace' => '3px']
  4590. );
  4591. }
  4592. // How many hours ?
  4593. $hours = floor($seconds / 3600);
  4594. // How many minutes ?
  4595. $min = floor(($seconds - ($hours * 3600)) / 60);
  4596. // How many seconds
  4597. $sec = floor($seconds - ($hours * 3600) - ($min * 60));
  4598. if ($hours < 10) {
  4599. $hours = "0$hours";
  4600. }
  4601. if ($sec < 10) {
  4602. $sec = "0$sec";
  4603. }
  4604. if ($min < 10) {
  4605. $min = "0$min";
  4606. }
  4607. return $hours.$space.$min.$space.$sec;
  4608. }
  4609. /* FILE SYSTEM RELATED FUNCTIONS */
  4610. /**
  4611. * Returns the permissions to be assigned to every newly created directory by the web-server.
  4612. * The return value is based on the platform administrator's setting
  4613. * "Administration > Configuration settings > Security > Permissions for new directories".
  4614. *
  4615. * @return int returns the permissions in the format "Owner-Group-Others, Read-Write-Execute", as an integer value
  4616. */
  4617. function api_get_permissions_for_new_directories()
  4618. {
  4619. static $permissions;
  4620. if (!isset($permissions)) {
  4621. $permissions = trim(api_get_setting('permissions_for_new_directories'));
  4622. // The default value 0777 is according to that in the platform administration panel after fresh system installation.
  4623. $permissions = octdec(!empty($permissions) ? $permissions : '0777');
  4624. }
  4625. return $permissions;
  4626. }
  4627. /**
  4628. * Returns the permissions to be assigned to every newly created directory by the web-server.
  4629. * The return value is based on the platform administrator's setting
  4630. * "Administration > Configuration settings > Security > Permissions for new files".
  4631. *
  4632. * @return int returns the permissions in the format
  4633. * "Owner-Group-Others, Read-Write-Execute", as an integer value
  4634. */
  4635. function api_get_permissions_for_new_files()
  4636. {
  4637. static $permissions;
  4638. if (!isset($permissions)) {
  4639. $permissions = trim(api_get_setting('permissions_for_new_files'));
  4640. // The default value 0666 is according to that in the platform
  4641. // administration panel after fresh system installation.
  4642. $permissions = octdec(!empty($permissions) ? $permissions : '0666');
  4643. }
  4644. return $permissions;
  4645. }
  4646. /**
  4647. * Deletes a file, or a folder and its contents.
  4648. *
  4649. * @author Aidan Lister <aidan@php.net>
  4650. *
  4651. * @version 1.0.3
  4652. *
  4653. * @param string $dirname Directory to delete
  4654. * @param bool Deletes only the content or not
  4655. * @param bool $strict if one folder/file fails stop the loop
  4656. *
  4657. * @return bool Returns TRUE on success, FALSE on failure
  4658. *
  4659. * @see http://aidanlister.com/2004/04/recursively-deleting-a-folder-in-php/
  4660. *
  4661. * @author Yannick Warnier, adaptation for the Chamilo LMS, April, 2008
  4662. * @author Ivan Tcholakov, a sanity check about Directory class creation has been added, September, 2009
  4663. */
  4664. function rmdirr($dirname, $delete_only_content_in_folder = false, $strict = false)
  4665. {
  4666. $res = true;
  4667. // A sanity check.
  4668. if (!file_exists($dirname)) {
  4669. return false;
  4670. }
  4671. $php_errormsg = '';
  4672. // Simple delete for a file.
  4673. if (is_file($dirname) || is_link($dirname)) {
  4674. $res = unlink($dirname);
  4675. if ($res === false) {
  4676. error_log(__FILE__.' line '.__LINE__.': '.((bool) ini_get('track_errors') ? $php_errormsg : 'Error not recorded because track_errors is off in your php.ini'), 0);
  4677. }
  4678. return $res;
  4679. }
  4680. // Loop through the folder.
  4681. $dir = dir($dirname);
  4682. // A sanity check.
  4683. $is_object_dir = is_object($dir);
  4684. if ($is_object_dir) {
  4685. while (false !== $entry = $dir->read()) {
  4686. // Skip pointers.
  4687. if ($entry == '.' || $entry == '..') {
  4688. continue;
  4689. }
  4690. // Recurse.
  4691. if ($strict) {
  4692. $result = rmdirr("$dirname/$entry");
  4693. if ($result == false) {
  4694. $res = false;
  4695. break;
  4696. }
  4697. } else {
  4698. rmdirr("$dirname/$entry");
  4699. }
  4700. }
  4701. }
  4702. // Clean up.
  4703. if ($is_object_dir) {
  4704. $dir->close();
  4705. }
  4706. if ($delete_only_content_in_folder == false) {
  4707. $res = rmdir($dirname);
  4708. if ($res === false) {
  4709. error_log(__FILE__.' line '.__LINE__.': '.((bool) ini_get('track_errors') ? $php_errormsg : 'error not recorded because track_errors is off in your php.ini'), 0);
  4710. }
  4711. }
  4712. return $res;
  4713. }
  4714. // TODO: This function is to be simplified. File access modes to be implemented.
  4715. /**
  4716. * function adapted from a php.net comment
  4717. * copy recursively a folder.
  4718. *
  4719. * @param the source folder
  4720. * @param the dest folder
  4721. * @param an array of excluded file_name (without extension)
  4722. * @param copied_files the returned array of copied files
  4723. * @param string $source
  4724. * @param string $dest
  4725. */
  4726. function copyr($source, $dest, $exclude = [], $copied_files = [])
  4727. {
  4728. if (empty($dest)) {
  4729. return false;
  4730. }
  4731. // Simple copy for a file
  4732. if (is_file($source)) {
  4733. $path_info = pathinfo($source);
  4734. if (!in_array($path_info['filename'], $exclude)) {
  4735. copy($source, $dest);
  4736. }
  4737. return true;
  4738. } elseif (!is_dir($source)) {
  4739. //then source is not a dir nor a file, return
  4740. return false;
  4741. }
  4742. // Make destination directory.
  4743. if (!is_dir($dest)) {
  4744. mkdir($dest, api_get_permissions_for_new_directories());
  4745. }
  4746. // Loop through the folder.
  4747. $dir = dir($source);
  4748. while (false !== $entry = $dir->read()) {
  4749. // Skip pointers
  4750. if ($entry == '.' || $entry == '..') {
  4751. continue;
  4752. }
  4753. // Deep copy directories.
  4754. if ($dest !== "$source/$entry") {
  4755. $files = copyr("$source/$entry", "$dest/$entry", $exclude, $copied_files);
  4756. }
  4757. }
  4758. // Clean up.
  4759. $dir->close();
  4760. return true;
  4761. }
  4762. /**
  4763. * @todo: Using DIRECTORY_SEPARATOR is not recommended, this is an obsolete approach.
  4764. * Documentation header to be added here.
  4765. *
  4766. * @param string $pathname
  4767. * @param string $base_path_document
  4768. * @param int $session_id
  4769. *
  4770. * @return mixed True if directory already exists, false if a file already exists at
  4771. * the destination and null if everything goes according to plan
  4772. */
  4773. function copy_folder_course_session(
  4774. $pathname,
  4775. $base_path_document,
  4776. $session_id,
  4777. $course_info,
  4778. $document,
  4779. $source_course_id
  4780. ) {
  4781. $table = Database::get_course_table(TABLE_DOCUMENT);
  4782. $session_id = intval($session_id);
  4783. $source_course_id = intval($source_course_id);
  4784. // Check whether directory already exists.
  4785. if (is_dir($pathname) || empty($pathname)) {
  4786. return true;
  4787. }
  4788. // Ensure that a file with the same name does not already exist.
  4789. if (is_file($pathname)) {
  4790. trigger_error('copy_folder_course_session(): File exists', E_USER_WARNING);
  4791. return false;
  4792. }
  4793. $course_id = $course_info['real_id'];
  4794. $folders = explode(DIRECTORY_SEPARATOR, str_replace($base_path_document.DIRECTORY_SEPARATOR, '', $pathname));
  4795. $new_pathname = $base_path_document;
  4796. $path = '';
  4797. foreach ($folders as $folder) {
  4798. $new_pathname .= DIRECTORY_SEPARATOR.$folder;
  4799. $path .= DIRECTORY_SEPARATOR.$folder;
  4800. if (!file_exists($new_pathname)) {
  4801. $path = Database::escape_string($path);
  4802. $sql = "SELECT * FROM $table
  4803. WHERE
  4804. c_id = $source_course_id AND
  4805. path = '$path' AND
  4806. filetype = 'folder' AND
  4807. session_id = '$session_id'";
  4808. $rs1 = Database::query($sql);
  4809. $num_rows = Database::num_rows($rs1);
  4810. if ($num_rows == 0) {
  4811. mkdir($new_pathname, api_get_permissions_for_new_directories());
  4812. // Insert new folder with destination session_id.
  4813. $params = [
  4814. 'c_id' => $course_id,
  4815. 'path' => $path,
  4816. 'comment' => $document->comment,
  4817. 'title' => basename($new_pathname),
  4818. 'filetype' => 'folder',
  4819. 'size' => '0',
  4820. 'session_id' => $session_id,
  4821. ];
  4822. $document_id = Database::insert($table, $params);
  4823. if ($document_id) {
  4824. $sql = "UPDATE $table SET id = iid WHERE iid = $document_id";
  4825. Database::query($sql);
  4826. api_item_property_update(
  4827. $course_info,
  4828. TOOL_DOCUMENT,
  4829. $document_id,
  4830. 'FolderCreated',
  4831. api_get_user_id(),
  4832. 0,
  4833. 0,
  4834. null,
  4835. null,
  4836. $session_id
  4837. );
  4838. }
  4839. }
  4840. }
  4841. } // en foreach
  4842. }
  4843. // TODO: chmodr() is a better name. Some corrections are needed. Documentation header to be added here.
  4844. /**
  4845. * @param string $path
  4846. */
  4847. function api_chmod_R($path, $filemode)
  4848. {
  4849. if (!is_dir($path)) {
  4850. return chmod($path, $filemode);
  4851. }
  4852. $handler = opendir($path);
  4853. while ($file = readdir($handler)) {
  4854. if ($file != '.' && $file != '..') {
  4855. $fullpath = "$path/$file";
  4856. if (!is_dir($fullpath)) {
  4857. if (!chmod($fullpath, $filemode)) {
  4858. return false;
  4859. }
  4860. } else {
  4861. if (!api_chmod_R($fullpath, $filemode)) {
  4862. return false;
  4863. }
  4864. }
  4865. }
  4866. }
  4867. closedir($handler);
  4868. return chmod($path, $filemode);
  4869. }
  4870. // TODO: Where the following function has been copy/pased from? There is no information about author and license. Style, coding conventions...
  4871. /**
  4872. * Parse info file format. (e.g: file.info).
  4873. *
  4874. * Files should use an ini-like format to specify values.
  4875. * White-space generally doesn't matter, except inside values.
  4876. * e.g.
  4877. *
  4878. * @verbatim
  4879. * key = value
  4880. * key = "value"
  4881. * key = 'value'
  4882. * key = "multi-line
  4883. *
  4884. * value"
  4885. * key = 'multi-line
  4886. *
  4887. * value'
  4888. * key
  4889. * =
  4890. * 'value'
  4891. * @endverbatim
  4892. *
  4893. * Arrays are created using a GET-like syntax:
  4894. *
  4895. * @verbatim
  4896. * key[] = "numeric array"
  4897. * key[index] = "associative array"
  4898. * key[index][] = "nested numeric array"
  4899. * key[index][index] = "nested associative array"
  4900. * @endverbatim
  4901. *
  4902. * PHP constants are substituted in, but only when used as the entire value:
  4903. *
  4904. * Comments should start with a semi-colon at the beginning of a line.
  4905. *
  4906. * This function is NOT for placing arbitrary module-specific settings. Use
  4907. * variable_get() and variable_set() for that.
  4908. *
  4909. * Information stored in the module.info file:
  4910. * - name: The real name of the module for display purposes.
  4911. * - description: A brief description of the module.
  4912. * - dependencies: An array of shortnames of other modules this module depends on.
  4913. * - package: The name of the package of modules this module belongs to.
  4914. *
  4915. * Example of .info file:
  4916. * <code>
  4917. * @verbatim
  4918. * name = Forum
  4919. * description = Enables threaded discussions about general topics.
  4920. * dependencies[] = taxonomy
  4921. * dependencies[] = comment
  4922. * package = Core - optional
  4923. * version = VERSION
  4924. * @endverbatim
  4925. * </code>
  4926. *
  4927. * @param string $filename
  4928. * The file we are parsing. Accepts file with relative or absolute path.
  4929. *
  4930. * @return
  4931. * The info array
  4932. */
  4933. function api_parse_info_file($filename)
  4934. {
  4935. $info = [];
  4936. if (!file_exists($filename)) {
  4937. return $info;
  4938. }
  4939. $data = file_get_contents($filename);
  4940. if (preg_match_all('
  4941. @^\s* # Start at the beginning of a line, ignoring leading whitespace
  4942. ((?:
  4943. [^=;\[\]]| # Key names cannot contain equal signs, semi-colons or square brackets,
  4944. \[[^\[\]]*\] # unless they are balanced and not nested
  4945. )+?)
  4946. \s*=\s* # Key/value pairs are separated by equal signs (ignoring white-space)
  4947. (?:
  4948. ("(?:[^"]|(?<=\\\\)")*")| # Double-quoted string, which may contain slash-escaped quotes/slashes
  4949. (\'(?:[^\']|(?<=\\\\)\')*\')| # Single-quoted string, which may contain slash-escaped quotes/slashes
  4950. ([^\r\n]*?) # Non-quoted string
  4951. )\s*$ # Stop at the next end of a line, ignoring trailing whitespace
  4952. @msx', $data, $matches, PREG_SET_ORDER)) {
  4953. $key = $value1 = $value2 = $value3 = '';
  4954. foreach ($matches as $match) {
  4955. // Fetch the key and value string.
  4956. $i = 0;
  4957. foreach (['key', 'value1', 'value2', 'value3'] as $var) {
  4958. $$var = isset($match[++$i]) ? $match[$i] : '';
  4959. }
  4960. $value = stripslashes(substr($value1, 1, -1)).stripslashes(substr($value2, 1, -1)).$value3;
  4961. // Parse array syntax.
  4962. $keys = preg_split('/\]?\[/', rtrim($key, ']'));
  4963. $last = array_pop($keys);
  4964. $parent = &$info;
  4965. // Create nested arrays.
  4966. foreach ($keys as $key) {
  4967. if ($key == '') {
  4968. $key = count($parent);
  4969. }
  4970. if (!isset($parent[$key]) || !is_array($parent[$key])) {
  4971. $parent[$key] = [];
  4972. }
  4973. $parent = &$parent[$key];
  4974. }
  4975. // Handle PHP constants.
  4976. if (defined($value)) {
  4977. $value = constant($value);
  4978. }
  4979. // Insert actual value.
  4980. if ($last == '') {
  4981. $last = count($parent);
  4982. }
  4983. $parent[$last] = $value;
  4984. }
  4985. }
  4986. return $info;
  4987. }
  4988. /**
  4989. * Gets Chamilo version from the configuration files.
  4990. *
  4991. * @return string A string of type "1.8.4", or an empty string if the version could not be found
  4992. */
  4993. function api_get_version()
  4994. {
  4995. return (string) api_get_configuration_value('system_version');
  4996. }
  4997. /**
  4998. * Gets the software name (the name/brand of the Chamilo-based customized system).
  4999. *
  5000. * @return string
  5001. */
  5002. function api_get_software_name()
  5003. {
  5004. $name = api_get_configuration_value('software_name');
  5005. if (!empty($name)) {
  5006. return $name;
  5007. } else {
  5008. return 'Chamilo';
  5009. }
  5010. }
  5011. /**
  5012. * Checks whether status given in parameter exists in the platform.
  5013. *
  5014. * @param mixed the status (can be either int either string)
  5015. *
  5016. * @return bool if the status exists, else returns false
  5017. */
  5018. function api_status_exists($status_asked)
  5019. {
  5020. global $_status_list;
  5021. return in_array($status_asked, $_status_list) ? true : isset($_status_list[$status_asked]);
  5022. }
  5023. /**
  5024. * Checks whether status given in parameter exists in the platform. The function
  5025. * returns the status ID or false if it does not exist, but given the fact there
  5026. * is no "0" status, the return value can be checked against
  5027. * if(api_status_key()) to know if it exists.
  5028. *
  5029. * @param mixed The status (can be either int or string)
  5030. *
  5031. * @return mixed Status ID if exists, false otherwise
  5032. */
  5033. function api_status_key($status)
  5034. {
  5035. global $_status_list;
  5036. return isset($_status_list[$status]) ? $status : array_search($status, $_status_list);
  5037. }
  5038. /**
  5039. * Gets the status langvars list.
  5040. *
  5041. * @return string[] the list of status with their translations
  5042. */
  5043. function api_get_status_langvars()
  5044. {
  5045. return [
  5046. COURSEMANAGER => get_lang('Teacher', ''),
  5047. SESSIONADMIN => get_lang('SessionsAdmin', ''),
  5048. DRH => get_lang('Drh', ''),
  5049. STUDENT => get_lang('Student', ''),
  5050. ANONYMOUS => get_lang('Anonymous', ''),
  5051. STUDENT_BOSS => get_lang('RoleStudentBoss', ''),
  5052. INVITEE => get_lang('Invited'),
  5053. ];
  5054. }
  5055. /**
  5056. * The function that retrieves all the possible settings for a certain config setting.
  5057. *
  5058. * @author Patrick Cool <patrick.cool@UGent.be>, Ghent University
  5059. */
  5060. function api_get_settings_options($var)
  5061. {
  5062. $table_settings_options = Database::get_main_table(TABLE_MAIN_SETTINGS_OPTIONS);
  5063. $var = Database::escape_string($var);
  5064. $sql = "SELECT * FROM $table_settings_options
  5065. WHERE variable = '$var'
  5066. ORDER BY id";
  5067. $result = Database::query($sql);
  5068. $settings_options_array = [];
  5069. while ($row = Database::fetch_array($result, 'ASSOC')) {
  5070. $settings_options_array[] = $row;
  5071. }
  5072. return $settings_options_array;
  5073. }
  5074. /**
  5075. * @param array $params
  5076. */
  5077. function api_set_setting_option($params)
  5078. {
  5079. $table = Database::get_main_table(TABLE_MAIN_SETTINGS_OPTIONS);
  5080. if (empty($params['id'])) {
  5081. Database::insert($table, $params);
  5082. } else {
  5083. Database::update($table, $params, ['id = ? ' => $params['id']]);
  5084. }
  5085. }
  5086. /**
  5087. * @param array $params
  5088. */
  5089. function api_set_setting_simple($params)
  5090. {
  5091. $table = Database::get_main_table(TABLE_MAIN_SETTINGS_CURRENT);
  5092. $url_id = api_get_current_access_url_id();
  5093. if (empty($params['id'])) {
  5094. $params['access_url'] = $url_id;
  5095. Database::insert($table, $params);
  5096. } else {
  5097. Database::update($table, $params, ['id = ? ' => [$params['id']]]);
  5098. }
  5099. }
  5100. /**
  5101. * @param int $id
  5102. */
  5103. function api_delete_setting_option($id)
  5104. {
  5105. $table = Database::get_main_table(TABLE_MAIN_SETTINGS_OPTIONS);
  5106. if (!empty($id)) {
  5107. Database::delete($table, ['id = ? ' => $id]);
  5108. }
  5109. }
  5110. /**
  5111. * Sets a platform configuration setting to a given value.
  5112. *
  5113. * @param string The variable we want to update
  5114. * @param string The value we want to record
  5115. * @param string The sub-variable if any (in most cases, this will remain null)
  5116. * @param string The category if any (in most cases, this will remain null)
  5117. * @param int The access_url for which this parameter is valid
  5118. * @param string $cat
  5119. *
  5120. * @return bool|null
  5121. */
  5122. function api_set_setting($var, $value, $subvar = null, $cat = null, $access_url = 1)
  5123. {
  5124. if (empty($var)) {
  5125. return false;
  5126. }
  5127. $t_settings = Database::get_main_table(TABLE_MAIN_SETTINGS_CURRENT);
  5128. $var = Database::escape_string($var);
  5129. $value = Database::escape_string($value);
  5130. $access_url = (int) $access_url;
  5131. if (empty($access_url)) {
  5132. $access_url = 1;
  5133. }
  5134. $select = "SELECT id FROM $t_settings WHERE variable = '$var' ";
  5135. if (!empty($subvar)) {
  5136. $subvar = Database::escape_string($subvar);
  5137. $select .= " AND subkey = '$subvar'";
  5138. }
  5139. if (!empty($cat)) {
  5140. $cat = Database::escape_string($cat);
  5141. $select .= " AND category = '$cat'";
  5142. }
  5143. if ($access_url > 1) {
  5144. $select .= " AND access_url = $access_url";
  5145. } else {
  5146. $select .= " AND access_url = 1 ";
  5147. }
  5148. $res = Database::query($select);
  5149. if (Database::num_rows($res) > 0) {
  5150. // Found item for this access_url.
  5151. $row = Database::fetch_array($res);
  5152. $sql = "UPDATE $t_settings SET selected_value = '$value'
  5153. WHERE id = ".$row['id'];
  5154. Database::query($sql);
  5155. } else {
  5156. // Item not found for this access_url, we have to check if it exist with access_url = 1
  5157. $select = "SELECT * FROM $t_settings
  5158. WHERE variable = '$var' AND access_url = 1 ";
  5159. // Just in case
  5160. if ($access_url == 1) {
  5161. if (!empty($subvar)) {
  5162. $select .= " AND subkey = '$subvar'";
  5163. }
  5164. if (!empty($cat)) {
  5165. $select .= " AND category = '$cat'";
  5166. }
  5167. $res = Database::query($select);
  5168. if (Database::num_rows($res) > 0) {
  5169. // We have a setting for access_url 1, but none for the current one, so create one.
  5170. $row = Database::fetch_array($res);
  5171. $insert = "INSERT INTO $t_settings (variable, subkey, type,category, selected_value, title, comment, scope, subkeytext, access_url)
  5172. VALUES
  5173. ('".$row['variable']."',".(!empty($row['subkey']) ? "'".$row['subkey']."'" : "NULL").",".
  5174. "'".$row['type']."','".$row['category']."',".
  5175. "'$value','".$row['title']."',".
  5176. "".(!empty($row['comment']) ? "'".$row['comment']."'" : "NULL").",".(!empty($row['scope']) ? "'".$row['scope']."'" : "NULL").",".
  5177. "".(!empty($row['subkeytext']) ? "'".$row['subkeytext']."'" : "NULL").",$access_url)";
  5178. Database::query($insert);
  5179. } else {
  5180. // Such a setting does not exist.
  5181. //error_log(__FILE__.':'.__LINE__.': Attempting to update setting '.$var.' ('.$subvar.') which does not exist at all', 0);
  5182. }
  5183. } else {
  5184. // Other access url.
  5185. if (!empty($subvar)) {
  5186. $select .= " AND subkey = '$subvar'";
  5187. }
  5188. if (!empty($cat)) {
  5189. $select .= " AND category = '$cat'";
  5190. }
  5191. $res = Database::query($select);
  5192. if (Database::num_rows($res) > 0) {
  5193. // We have a setting for access_url 1, but none for the current one, so create one.
  5194. $row = Database::fetch_array($res);
  5195. if ($row['access_url_changeable'] == 1) {
  5196. $insert = "INSERT INTO $t_settings (variable,subkey, type,category, selected_value,title, comment,scope, subkeytext,access_url, access_url_changeable) VALUES
  5197. ('".$row['variable']."',".
  5198. (!empty($row['subkey']) ? "'".$row['subkey']."'" : "NULL").",".
  5199. "'".$row['type']."','".$row['category']."',".
  5200. "'$value','".$row['title']."',".
  5201. "".(!empty($row['comment']) ? "'".$row['comment']."'" : "NULL").",".
  5202. (!empty($row['scope']) ? "'".$row['scope']."'" : "NULL").",".
  5203. "".(!empty($row['subkeytext']) ? "'".$row['subkeytext']."'" : "NULL").",$access_url,".$row['access_url_changeable'].")";
  5204. Database::query($insert);
  5205. }
  5206. } else { // Such a setting does not exist.
  5207. //error_log(__FILE__.':'.__LINE__.': Attempting to update setting '.$var.' ('.$subvar.') which does not exist at all. The access_url is: '.$access_url.' ',0);
  5208. }
  5209. }
  5210. }
  5211. }
  5212. /**
  5213. * Sets a whole category of settings to one specific value.
  5214. *
  5215. * @param string Category
  5216. * @param string Value
  5217. * @param int Access URL. Optional. Defaults to 1
  5218. * @param array Optional array of filters on field type
  5219. * @param string $category
  5220. * @param string $value
  5221. *
  5222. * @return bool
  5223. */
  5224. function api_set_settings_category($category, $value = null, $access_url = 1, $fieldtype = [])
  5225. {
  5226. if (empty($category)) {
  5227. return false;
  5228. }
  5229. $category = Database::escape_string($category);
  5230. $t_s = Database::get_main_table(TABLE_MAIN_SETTINGS_CURRENT);
  5231. $access_url = (int) $access_url;
  5232. if (empty($access_url)) {
  5233. $access_url = 1;
  5234. }
  5235. if (isset($value)) {
  5236. $value = Database::escape_string($value);
  5237. $sql = "UPDATE $t_s SET selected_value = '$value'
  5238. WHERE category = '$category' AND access_url = $access_url";
  5239. if (is_array($fieldtype) && count($fieldtype) > 0) {
  5240. $sql .= " AND ( ";
  5241. $i = 0;
  5242. foreach ($fieldtype as $type) {
  5243. if ($i > 0) {
  5244. $sql .= ' OR ';
  5245. }
  5246. $type = Database::escape_string($type);
  5247. $sql .= " type='".$type."' ";
  5248. $i++;
  5249. }
  5250. $sql .= ")";
  5251. }
  5252. $res = Database::query($sql);
  5253. return $res !== false;
  5254. } else {
  5255. $sql = "UPDATE $t_s SET selected_value = NULL
  5256. WHERE category = '$category' AND access_url = $access_url";
  5257. if (is_array($fieldtype) && count($fieldtype) > 0) {
  5258. $sql .= " AND ( ";
  5259. $i = 0;
  5260. foreach ($fieldtype as $type) {
  5261. if ($i > 0) {
  5262. $sql .= ' OR ';
  5263. }
  5264. $type = Database::escape_string($type);
  5265. $sql .= " type='".$type."' ";
  5266. $i++;
  5267. }
  5268. $sql .= ")";
  5269. }
  5270. $res = Database::query($sql);
  5271. return $res !== false;
  5272. }
  5273. }
  5274. /**
  5275. * Gets all available access urls in an array (as in the database).
  5276. *
  5277. * @return array An array of database records
  5278. */
  5279. function api_get_access_urls($from = 0, $to = 1000000, $order = 'url', $direction = 'ASC')
  5280. {
  5281. $table = Database::get_main_table(TABLE_MAIN_ACCESS_URL);
  5282. $from = (int) $from;
  5283. $to = (int) $to;
  5284. $order = Database::escape_string($order, null, false);
  5285. $direction = Database::escape_string($direction, null, false);
  5286. $sql = "SELECT id, url, description, active, created_by, tms
  5287. FROM $table
  5288. ORDER BY $order $direction
  5289. LIMIT $to OFFSET $from";
  5290. $res = Database::query($sql);
  5291. return Database::store_result($res);
  5292. }
  5293. /**
  5294. * Gets the access url info in an array.
  5295. *
  5296. * @param int $id Id of the access url
  5297. * @param bool $returnDefault Set to false if you want the real URL if URL 1 is still 'http://localhost/'
  5298. *
  5299. * @return array All the info (url, description, active, created_by, tms)
  5300. * from the access_url table
  5301. *
  5302. * @author Julio Montoya
  5303. */
  5304. function api_get_access_url($id, $returnDefault = true)
  5305. {
  5306. static $staticResult;
  5307. $id = (int) $id;
  5308. if (isset($staticResult[$id])) {
  5309. $result = $staticResult[$id];
  5310. } else {
  5311. // Calling the Database:: library dont work this is handmade.
  5312. $table_access_url = Database::get_main_table(TABLE_MAIN_ACCESS_URL);
  5313. $sql = "SELECT url, description, active, created_by, tms
  5314. FROM $table_access_url WHERE id = '$id' ";
  5315. $res = Database::query($sql);
  5316. $result = @Database::fetch_array($res);
  5317. $staticResult[$id] = $result;
  5318. }
  5319. // If the result url is 'http://localhost/' (the default) and the root_web
  5320. // (=current url) is different, and the $id is = 1 (which might mean
  5321. // api_get_current_access_url_id() returned 1 by default), then return the
  5322. // root_web setting instead of the current URL
  5323. // This is provided as an option to avoid breaking the storage of URL-specific
  5324. // homepages in home/localhost/
  5325. if ($id === 1 && $returnDefault === false) {
  5326. $currentUrl = api_get_current_access_url_id();
  5327. // only do this if we are on the main URL (=1), otherwise we could get
  5328. // information on another URL instead of the one asked as parameter
  5329. if ($currentUrl === 1) {
  5330. $rootWeb = api_get_path(WEB_PATH);
  5331. $default = 'http://localhost/';
  5332. if ($result['url'] === $default && $rootWeb != $default) {
  5333. $result['url'] = $rootWeb;
  5334. }
  5335. }
  5336. }
  5337. return $result;
  5338. }
  5339. /**
  5340. * Gets all the current settings for a specific access url.
  5341. *
  5342. * @param string The category, if any, that we want to get
  5343. * @param string Whether we want a simple list (display a category) or
  5344. * a grouped list (group by variable as in settings.php default). Values: 'list' or 'group'
  5345. * @param int Access URL's ID. Optional. Uses 1 by default, which is the unique URL
  5346. *
  5347. * @return array Array of database results for the current settings of the current access URL
  5348. */
  5349. function &api_get_settings($cat = null, $ordering = 'list', $access_url = 1, $url_changeable = 0)
  5350. {
  5351. $table = Database::get_main_table(TABLE_MAIN_SETTINGS_CURRENT);
  5352. $access_url = (int) $access_url;
  5353. $where_condition = '';
  5354. if ($url_changeable == 1) {
  5355. $where_condition = " AND access_url_changeable= '1' ";
  5356. }
  5357. if (empty($access_url) || $access_url == -1) {
  5358. $access_url = 1;
  5359. }
  5360. $sql = "SELECT * FROM $table
  5361. WHERE access_url = $access_url $where_condition ";
  5362. if (!empty($cat)) {
  5363. $cat = Database::escape_string($cat);
  5364. $sql .= " AND category='$cat' ";
  5365. }
  5366. if ($ordering == 'group') {
  5367. $sql .= " ORDER BY id ASC";
  5368. } else {
  5369. $sql .= " ORDER BY 1,2 ASC";
  5370. }
  5371. $result = Database::query($sql);
  5372. if ($result === null) {
  5373. return [];
  5374. }
  5375. $result = Database::store_result($result, 'ASSOC');
  5376. return $result;
  5377. }
  5378. /**
  5379. * @param string $value The value we want to record
  5380. * @param string $variable The variable name we want to insert
  5381. * @param string $subKey The subkey for the variable we want to insert
  5382. * @param string $type The type for the variable we want to insert
  5383. * @param string $category The category for the variable we want to insert
  5384. * @param string $title The title
  5385. * @param string $comment The comment
  5386. * @param string $scope The scope
  5387. * @param string $subKeyText The subkey text
  5388. * @param int $accessUrlId The access_url for which this parameter is valid
  5389. * @param int $visibility The changeability of this setting for non-master urls
  5390. *
  5391. * @return int The setting ID
  5392. */
  5393. function api_add_setting(
  5394. $value,
  5395. $variable,
  5396. $subKey = '',
  5397. $type = 'textfield',
  5398. $category = '',
  5399. $title = '',
  5400. $comment = '',
  5401. $scope = '',
  5402. $subKeyText = '',
  5403. $accessUrlId = 1,
  5404. $visibility = 0
  5405. ) {
  5406. $em = Database::getManager();
  5407. $settingRepo = $em->getRepository('ChamiloCoreBundle:SettingsCurrent');
  5408. $accessUrlId = (int) $accessUrlId ?: 1;
  5409. if (is_array($value)) {
  5410. $value = serialize($value);
  5411. } else {
  5412. $value = trim($value);
  5413. }
  5414. $criteria = ['variable' => $variable, 'accessUrl' => $accessUrlId];
  5415. if (!empty($subKey)) {
  5416. $criteria['subkey'] = $subKey;
  5417. }
  5418. // Check if this variable doesn't exist already
  5419. /** @var SettingsCurrent $setting */
  5420. $setting = $settingRepo->findOneBy($criteria);
  5421. if ($setting) {
  5422. $setting->setSelectedValue($value);
  5423. $em->persist($setting);
  5424. $em->flush();
  5425. return $setting->getId();
  5426. }
  5427. // Item not found for this access_url, we have to check if the whole thing is missing
  5428. // (in which case we ignore the insert) or if there *is* a record but just for access_url = 1
  5429. $setting = new SettingsCurrent();
  5430. $setting
  5431. ->setVariable($variable)
  5432. ->setSelectedValue($value)
  5433. ->setType($type)
  5434. ->setCategory($category)
  5435. ->setSubkey($subKey)
  5436. ->setTitle($title)
  5437. ->setComment($comment)
  5438. ->setScope($scope)
  5439. ->setSubkeytext($subKeyText)
  5440. ->setAccessUrl($accessUrlId)
  5441. ->setAccessUrlChangeable($visibility);
  5442. $em->persist($setting);
  5443. $em->flush();
  5444. return $setting->getId();
  5445. }
  5446. /**
  5447. * Checks wether a user can or can't view the contents of a course.
  5448. *
  5449. * @deprecated use CourseManager::is_user_subscribed_in_course
  5450. *
  5451. * @param int $userid User id or NULL to get it from $_SESSION
  5452. * @param int $cid course id to check whether the user is allowed
  5453. *
  5454. * @return bool
  5455. */
  5456. function api_is_course_visible_for_user($userid = null, $cid = null)
  5457. {
  5458. if ($userid === null) {
  5459. $userid = api_get_user_id();
  5460. }
  5461. if (empty($userid) || strval(intval($userid)) != $userid) {
  5462. if (api_is_anonymous()) {
  5463. $userid = api_get_anonymous_id();
  5464. } else {
  5465. return false;
  5466. }
  5467. }
  5468. $cid = Database::escape_string($cid);
  5469. $courseInfo = api_get_course_info($cid);
  5470. $courseId = $courseInfo['real_id'];
  5471. $is_platformAdmin = api_is_platform_admin();
  5472. $course_table = Database::get_main_table(TABLE_MAIN_COURSE);
  5473. $course_cat_table = Database::get_main_table(TABLE_MAIN_CATEGORY);
  5474. $sql = "SELECT
  5475. $course_table.category_code,
  5476. $course_table.visibility,
  5477. $course_table.code,
  5478. $course_cat_table.code
  5479. FROM $course_table
  5480. LEFT JOIN $course_cat_table
  5481. ON $course_table.category_code = $course_cat_table.code
  5482. WHERE
  5483. $course_table.code = '$cid'
  5484. LIMIT 1";
  5485. $result = Database::query($sql);
  5486. if (Database::num_rows($result) > 0) {
  5487. $visibility = Database::fetch_array($result);
  5488. $visibility = $visibility['visibility'];
  5489. } else {
  5490. $visibility = 0;
  5491. }
  5492. // Shortcut permissions in case the visibility is "open to the world".
  5493. if ($visibility === COURSE_VISIBILITY_OPEN_WORLD) {
  5494. return true;
  5495. }
  5496. $tbl_course_user = Database::get_main_table(TABLE_MAIN_COURSE_USER);
  5497. $sql = "SELECT
  5498. is_tutor, status
  5499. FROM $tbl_course_user
  5500. WHERE
  5501. user_id = '$userid' AND
  5502. relation_type <> '".COURSE_RELATION_TYPE_RRHH."' AND
  5503. c_id = $courseId
  5504. LIMIT 1";
  5505. $result = Database::query($sql);
  5506. if (Database::num_rows($result) > 0) {
  5507. // This user has got a recorded state for this course.
  5508. $cuData = Database::fetch_array($result);
  5509. $is_courseMember = true;
  5510. $is_courseAdmin = ($cuData['status'] == 1);
  5511. }
  5512. if (!$is_courseAdmin) {
  5513. // This user has no status related to this course.
  5514. // Is it the session coach or the session admin?
  5515. $tbl_session = Database::get_main_table(TABLE_MAIN_SESSION);
  5516. $tbl_session_course = Database::get_main_table(TABLE_MAIN_SESSION_COURSE);
  5517. $tbl_session_course_user = Database::get_main_table(TABLE_MAIN_SESSION_COURSE_USER);
  5518. $sql = "SELECT
  5519. session.id_coach, session_admin_id, session.id
  5520. FROM
  5521. $tbl_session as session
  5522. INNER JOIN $tbl_session_course
  5523. ON session_rel_course.session_id = session.id
  5524. AND session_rel_course.c_id = '$courseId'
  5525. LIMIT 1";
  5526. $result = Database::query($sql);
  5527. $row = Database::store_result($result);
  5528. if ($row[0]['id_coach'] == $userid) {
  5529. $is_courseMember = true;
  5530. $is_courseAdmin = false;
  5531. } elseif ($row[0]['session_admin_id'] == $userid) {
  5532. $is_courseMember = false;
  5533. $is_courseAdmin = false;
  5534. } else {
  5535. // Check if the current user is the course coach.
  5536. $sql = "SELECT 1
  5537. FROM $tbl_session_course
  5538. WHERE session_rel_course.c_id = '$courseId'
  5539. AND session_rel_course.id_coach = '$userid'
  5540. LIMIT 1";
  5541. $result = Database::query($sql);
  5542. //if ($row = Database::fetch_array($result)) {
  5543. if (Database::num_rows($result) > 0) {
  5544. $is_courseMember = true;
  5545. $tbl_user = Database::get_main_table(TABLE_MAIN_USER);
  5546. $sql = "SELECT status FROM $tbl_user
  5547. WHERE user_id = $userid
  5548. LIMIT 1";
  5549. $result = Database::query($sql);
  5550. if (Database::result($result, 0, 0) == 1) {
  5551. $is_courseAdmin = true;
  5552. } else {
  5553. $is_courseAdmin = false;
  5554. }
  5555. } else {
  5556. // Check if the user is a student is this session.
  5557. $sql = "SELECT id
  5558. FROM $tbl_session_course_user
  5559. WHERE
  5560. user_id = '$userid' AND
  5561. c_id = '$courseId'
  5562. LIMIT 1";
  5563. if (Database::num_rows($result) > 0) {
  5564. // This user haa got a recorded state for this course.
  5565. while ($row = Database::fetch_array($result)) {
  5566. $is_courseMember = true;
  5567. $is_courseAdmin = false;
  5568. }
  5569. }
  5570. }
  5571. }
  5572. }
  5573. switch ($visibility) {
  5574. case COURSE_VISIBILITY_OPEN_WORLD:
  5575. return true;
  5576. case COURSE_VISIBILITY_OPEN_PLATFORM:
  5577. return isset($userid);
  5578. case COURSE_VISIBILITY_REGISTERED:
  5579. case COURSE_VISIBILITY_CLOSED:
  5580. return $is_platformAdmin || $is_courseMember || $is_courseAdmin;
  5581. case COURSE_VISIBILITY_HIDDEN:
  5582. return $is_platformAdmin;
  5583. }
  5584. return false;
  5585. }
  5586. /**
  5587. * Returns whether an element (forum, message, survey ...) belongs to a session or not.
  5588. *
  5589. * @param string the tool of the element
  5590. * @param int the element id in database
  5591. * @param int the session_id to compare with element session id
  5592. *
  5593. * @return bool true if the element is in the session, false else
  5594. */
  5595. function api_is_element_in_the_session($tool, $element_id, $session_id = null)
  5596. {
  5597. if (is_null($session_id)) {
  5598. $session_id = api_get_session_id();
  5599. }
  5600. $element_id = (int) $element_id;
  5601. if (empty($element_id)) {
  5602. return false;
  5603. }
  5604. // Get information to build query depending of the tool.
  5605. switch ($tool) {
  5606. case TOOL_SURVEY:
  5607. $table_tool = Database::get_course_table(TABLE_SURVEY);
  5608. $key_field = 'survey_id';
  5609. break;
  5610. case TOOL_ANNOUNCEMENT:
  5611. $table_tool = Database::get_course_table(TABLE_ANNOUNCEMENT);
  5612. $key_field = 'id';
  5613. break;
  5614. case TOOL_AGENDA:
  5615. $table_tool = Database::get_course_table(TABLE_AGENDA);
  5616. $key_field = 'id';
  5617. break;
  5618. case TOOL_GROUP:
  5619. $table_tool = Database::get_course_table(TABLE_GROUP);
  5620. $key_field = 'id';
  5621. break;
  5622. default:
  5623. return false;
  5624. }
  5625. $course_id = api_get_course_int_id();
  5626. $sql = "SELECT session_id FROM $table_tool
  5627. WHERE c_id = $course_id AND $key_field = ".$element_id;
  5628. $rs = Database::query($sql);
  5629. if ($element_session_id = Database::result($rs, 0, 0)) {
  5630. if ($element_session_id == intval($session_id)) {
  5631. // The element belongs to the session.
  5632. return true;
  5633. }
  5634. }
  5635. return false;
  5636. }
  5637. /**
  5638. * Replaces "forbidden" characters in a filename string.
  5639. *
  5640. * @param string $filename
  5641. * @param bool $treat_spaces_as_hyphens
  5642. *
  5643. * @return string
  5644. */
  5645. function api_replace_dangerous_char($filename, $treat_spaces_as_hyphens = true)
  5646. {
  5647. // Some non-properly encoded file names can cause the whole file to be
  5648. // skipped when uploaded. Avoid this by detecting the encoding and
  5649. // converting to UTF-8, setting the source as ASCII (a reasonably
  5650. // limited characters set) if nothing could be found (BT#
  5651. $encoding = api_detect_encoding($filename);
  5652. if (empty($encoding)) {
  5653. $encoding = 'ASCII';
  5654. if (!api_is_valid_ascii($filename)) {
  5655. // try iconv and try non standard ASCII a.k.a CP437
  5656. // see BT#15022
  5657. if (function_exists('iconv')) {
  5658. $result = iconv('CP437', 'UTF-8', $filename);
  5659. if (api_is_valid_utf8($result)) {
  5660. $filename = $result;
  5661. $encoding = 'UTF-8';
  5662. }
  5663. }
  5664. }
  5665. }
  5666. $filename = api_to_system_encoding($filename, $encoding);
  5667. $url = URLify::filter(
  5668. $filename,
  5669. 250,
  5670. '',
  5671. true,
  5672. true,
  5673. false,
  5674. false,
  5675. $treat_spaces_as_hyphens
  5676. );
  5677. return $url;
  5678. }
  5679. /**
  5680. * Fixes the $_SERVER['REQUEST_URI'] that is empty in IIS6.
  5681. *
  5682. * @author Ivan Tcholakov, 28-JUN-2006.
  5683. */
  5684. function api_request_uri()
  5685. {
  5686. if (!empty($_SERVER['REQUEST_URI'])) {
  5687. return $_SERVER['REQUEST_URI'];
  5688. }
  5689. $uri = $_SERVER['SCRIPT_NAME'];
  5690. if (!empty($_SERVER['QUERY_STRING'])) {
  5691. $uri .= '?'.$_SERVER['QUERY_STRING'];
  5692. }
  5693. $_SERVER['REQUEST_URI'] = $uri;
  5694. return $uri;
  5695. }
  5696. /** Gets the current access_url id of the Chamilo Platform
  5697. * @author Julio Montoya <gugli100@gmail.com>
  5698. *
  5699. * @return int access_url_id of the current Chamilo Installation
  5700. */
  5701. function api_get_current_access_url_id()
  5702. {
  5703. $access_url_table = Database::get_main_table(TABLE_MAIN_ACCESS_URL);
  5704. $path = Database::escape_string(api_get_path(WEB_PATH));
  5705. $sql = "SELECT id FROM $access_url_table WHERE url = '".$path."'";
  5706. $result = Database::query($sql);
  5707. if (Database::num_rows($result) > 0) {
  5708. $access_url_id = Database::result($result, 0, 0);
  5709. if ($access_url_id === false) {
  5710. return -1;
  5711. }
  5712. return (int) $access_url_id;
  5713. }
  5714. //if the url in WEB_PATH was not found, it can only mean that there is
  5715. // either a configuration problem or the first URL has not been defined yet
  5716. // (by default it is http://localhost/). Thus the more sensible thing we can
  5717. // do is return 1 (the main URL) as the user cannot hack this value anyway
  5718. return 1;
  5719. }
  5720. /**
  5721. * Gets the registered urls from a given user id.
  5722. *
  5723. * @author Julio Montoya <gugli100@gmail.com>
  5724. *
  5725. * @return int user id
  5726. */
  5727. function api_get_access_url_from_user($user_id)
  5728. {
  5729. $user_id = (int) $user_id;
  5730. $table_url_rel_user = Database::get_main_table(TABLE_MAIN_ACCESS_URL_REL_USER);
  5731. $table_url = Database::get_main_table(TABLE_MAIN_ACCESS_URL);
  5732. $sql = "SELECT access_url_id
  5733. FROM $table_url_rel_user url_rel_user
  5734. INNER JOIN $table_url u
  5735. ON (url_rel_user.access_url_id = u.id)
  5736. WHERE user_id = ".intval($user_id);
  5737. $result = Database::query($sql);
  5738. $list = [];
  5739. while ($row = Database::fetch_array($result, 'ASSOC')) {
  5740. $list[] = $row['access_url_id'];
  5741. }
  5742. return $list;
  5743. }
  5744. /**
  5745. * Gets the status of a user in a course.
  5746. *
  5747. * @param int $user_id
  5748. * @param int $courseId
  5749. *
  5750. * @return int user status
  5751. */
  5752. function api_get_status_of_user_in_course($user_id, $courseId)
  5753. {
  5754. $tbl_rel_course_user = Database::get_main_table(TABLE_MAIN_COURSE_USER);
  5755. if (!empty($user_id) && !empty($courseId)) {
  5756. $user_id = intval($user_id);
  5757. $courseId = intval($courseId);
  5758. $sql = 'SELECT status
  5759. FROM '.$tbl_rel_course_user.'
  5760. WHERE user_id='.$user_id.' AND c_id = '.$courseId;
  5761. $result = Database::query($sql);
  5762. $row_status = Database::fetch_array($result, 'ASSOC');
  5763. return $row_status['status'];
  5764. } else {
  5765. return 0;
  5766. }
  5767. }
  5768. /**
  5769. * Checks whether the curent user is in a group or not.
  5770. *
  5771. * @param string The group id - optional (takes it from session if not given)
  5772. * @param string The course code - optional (no additional check by course if course code is not given)
  5773. *
  5774. * @return bool
  5775. *
  5776. * @author Ivan Tcholakov
  5777. */
  5778. function api_is_in_group($groupIdParam = null, $courseCodeParam = null)
  5779. {
  5780. if (!empty($courseCodeParam)) {
  5781. $courseCode = api_get_course_id();
  5782. if (!empty($courseCode)) {
  5783. if ($courseCodeParam != $courseCode) {
  5784. return false;
  5785. }
  5786. } else {
  5787. return false;
  5788. }
  5789. }
  5790. $groupId = api_get_group_id();
  5791. if (isset($groupId) && $groupId != '') {
  5792. if (!empty($groupIdParam)) {
  5793. return $groupIdParam == $groupId;
  5794. } else {
  5795. return true;
  5796. }
  5797. }
  5798. return false;
  5799. }
  5800. /**
  5801. * Checks whether a secret key is valid.
  5802. *
  5803. * @param string $original_key_secret - secret key from (webservice) client
  5804. * @param string $security_key - security key from Chamilo
  5805. *
  5806. * @return bool - true if secret key is valid, false otherwise
  5807. */
  5808. function api_is_valid_secret_key($original_key_secret, $security_key)
  5809. {
  5810. return $original_key_secret == sha1($security_key);
  5811. }
  5812. /**
  5813. * Checks whether a user is into course.
  5814. *
  5815. * @param int $course_id - the course id
  5816. * @param int $user_id - the user id
  5817. *
  5818. * @return bool
  5819. */
  5820. function api_is_user_of_course($course_id, $user_id)
  5821. {
  5822. $tbl_course_rel_user = Database::get_main_table(TABLE_MAIN_COURSE_USER);
  5823. $sql = 'SELECT user_id FROM '.$tbl_course_rel_user.'
  5824. WHERE
  5825. c_id ="'.intval($course_id).'" AND
  5826. user_id = "'.intval($user_id).'" AND
  5827. relation_type <> '.COURSE_RELATION_TYPE_RRHH.' ';
  5828. $result = Database::query($sql);
  5829. return Database::num_rows($result) == 1;
  5830. }
  5831. /**
  5832. * Checks whether the server's operating system is Windows (TM).
  5833. *
  5834. * @return bool - true if the operating system is Windows, false otherwise
  5835. */
  5836. function api_is_windows_os()
  5837. {
  5838. if (function_exists('php_uname')) {
  5839. // php_uname() exists as of PHP 4.0.2, according to the documentation.
  5840. // We expect that this function will always work for Chamilo 1.8.x.
  5841. $os = php_uname();
  5842. }
  5843. // The following methods are not needed, but let them stay, just in case.
  5844. elseif (isset($_ENV['OS'])) {
  5845. // Sometimes $_ENV['OS'] may not be present (bugs?)
  5846. $os = $_ENV['OS'];
  5847. } elseif (defined('PHP_OS')) {
  5848. // PHP_OS means on which OS PHP was compiled, this is why
  5849. // using PHP_OS is the last choice for detection.
  5850. $os = PHP_OS;
  5851. } else {
  5852. return false;
  5853. }
  5854. return strtolower(substr((string) $os, 0, 3)) == 'win';
  5855. }
  5856. /**
  5857. * This function informs whether the sent request is XMLHttpRequest.
  5858. */
  5859. function api_is_xml_http_request()
  5860. {
  5861. return isset($_SERVER['HTTP_X_REQUESTED_WITH']) && strtolower($_SERVER['HTTP_X_REQUESTED_WITH']) == 'xmlhttprequest';
  5862. }
  5863. /**
  5864. * This wrapper function has been implemented for avoiding some known problems about the function getimagesize().
  5865. *
  5866. * @see http://php.net/manual/en/function.getimagesize.php
  5867. * @see http://www.dokeos.com/forum/viewtopic.php?t=12345
  5868. * @see http://www.dokeos.com/forum/viewtopic.php?t=16355
  5869. *
  5870. * @return int
  5871. */
  5872. function api_getimagesize($path)
  5873. {
  5874. $image = new Image($path);
  5875. return $image->get_image_size();
  5876. }
  5877. /**
  5878. * This function resizes an image, with preserving its proportions (or aspect ratio).
  5879. *
  5880. * @author Ivan Tcholakov, MAY-2009.
  5881. *
  5882. * @param int $image System path or URL of the image
  5883. * @param int $target_width Targeted width
  5884. * @param int $target_height Targeted height
  5885. *
  5886. * @return array Calculated new width and height
  5887. */
  5888. function api_resize_image($image, $target_width, $target_height)
  5889. {
  5890. $image_properties = api_getimagesize($image);
  5891. return api_calculate_image_size(
  5892. $image_properties['width'],
  5893. $image_properties['height'],
  5894. $target_width,
  5895. $target_height
  5896. );
  5897. }
  5898. /**
  5899. * This function calculates new image size, with preserving image's proportions (or aspect ratio).
  5900. *
  5901. * @author Ivan Tcholakov, MAY-2009.
  5902. * @author The initial idea has been taken from code by Patrick Cool, MAY-2004.
  5903. *
  5904. * @param int $image_width Initial width
  5905. * @param int $image_height Initial height
  5906. * @param int $target_width Targeted width
  5907. * @param int $target_height Targeted height
  5908. *
  5909. * @return array Calculated new width and height
  5910. */
  5911. function api_calculate_image_size(
  5912. $image_width,
  5913. $image_height,
  5914. $target_width,
  5915. $target_height
  5916. ) {
  5917. // Only maths is here.
  5918. $result = ['width' => $image_width, 'height' => $image_height];
  5919. if ($image_width <= 0 || $image_height <= 0) {
  5920. return $result;
  5921. }
  5922. $resize_factor_width = $target_width / $image_width;
  5923. $resize_factor_height = $target_height / $image_height;
  5924. $delta_width = $target_width - $image_width * $resize_factor_height;
  5925. $delta_height = $target_height - $image_height * $resize_factor_width;
  5926. if ($delta_width > $delta_height) {
  5927. $result['width'] = ceil($image_width * $resize_factor_height);
  5928. $result['height'] = ceil($image_height * $resize_factor_height);
  5929. } elseif ($delta_width < $delta_height) {
  5930. $result['width'] = ceil($image_width * $resize_factor_width);
  5931. $result['height'] = ceil($image_height * $resize_factor_width);
  5932. } else {
  5933. $result['width'] = ceil($target_width);
  5934. $result['height'] = ceil($target_height);
  5935. }
  5936. return $result;
  5937. }
  5938. /**
  5939. * Returns a list of Chamilo's tools or
  5940. * checks whether a given identificator is a valid Chamilo's tool.
  5941. *
  5942. * @author Isaac flores paz
  5943. *
  5944. * @param string The tool name to filter
  5945. *
  5946. * @return mixed Filtered string or array
  5947. */
  5948. function api_get_tools_lists($my_tool = null)
  5949. {
  5950. $tools_list = [
  5951. TOOL_DOCUMENT,
  5952. TOOL_THUMBNAIL,
  5953. TOOL_HOTPOTATOES,
  5954. TOOL_CALENDAR_EVENT,
  5955. TOOL_LINK,
  5956. TOOL_COURSE_DESCRIPTION,
  5957. TOOL_SEARCH,
  5958. TOOL_LEARNPATH,
  5959. TOOL_ANNOUNCEMENT,
  5960. TOOL_FORUM,
  5961. TOOL_THREAD,
  5962. TOOL_POST,
  5963. TOOL_DROPBOX,
  5964. TOOL_QUIZ,
  5965. TOOL_USER,
  5966. TOOL_GROUP,
  5967. TOOL_BLOGS,
  5968. TOOL_CHAT,
  5969. TOOL_STUDENTPUBLICATION,
  5970. TOOL_TRACKING,
  5971. TOOL_HOMEPAGE_LINK,
  5972. TOOL_COURSE_SETTING,
  5973. TOOL_BACKUP,
  5974. TOOL_COPY_COURSE_CONTENT,
  5975. TOOL_RECYCLE_COURSE,
  5976. TOOL_COURSE_HOMEPAGE,
  5977. TOOL_COURSE_RIGHTS_OVERVIEW,
  5978. TOOL_UPLOAD,
  5979. TOOL_COURSE_MAINTENANCE,
  5980. TOOL_SURVEY,
  5981. TOOL_WIKI,
  5982. TOOL_GLOSSARY,
  5983. TOOL_GRADEBOOK,
  5984. TOOL_NOTEBOOK,
  5985. TOOL_ATTENDANCE,
  5986. TOOL_COURSE_PROGRESS,
  5987. ];
  5988. if (empty($my_tool)) {
  5989. return $tools_list;
  5990. }
  5991. return in_array($my_tool, $tools_list) ? $my_tool : '';
  5992. }
  5993. /**
  5994. * Checks whether we already approved the last version term and condition.
  5995. *
  5996. * @param int user id
  5997. *
  5998. * @return bool true if we pass false otherwise
  5999. */
  6000. function api_check_term_condition($userId)
  6001. {
  6002. if (api_get_setting('allow_terms_conditions') === 'true') {
  6003. // Check if exists terms and conditions
  6004. if (LegalManager::count() == 0) {
  6005. return true;
  6006. }
  6007. $extraFieldValue = new ExtraFieldValue('user');
  6008. $data = $extraFieldValue->get_values_by_handler_and_field_variable(
  6009. $userId,
  6010. 'legal_accept'
  6011. );
  6012. if (!empty($data) && isset($data['value']) && !empty($data['value'])) {
  6013. $result = $data['value'];
  6014. $user_conditions = explode(':', $result);
  6015. $version = $user_conditions[0];
  6016. $langId = $user_conditions[1];
  6017. $realVersion = LegalManager::get_last_version($langId);
  6018. return $version >= $realVersion;
  6019. }
  6020. return false;
  6021. }
  6022. return false;
  6023. }
  6024. /**
  6025. * Gets all information of a tool into course.
  6026. *
  6027. * @param int The tool id
  6028. *
  6029. * @return array
  6030. */
  6031. function api_get_tool_information_by_name($name)
  6032. {
  6033. $t_tool = Database::get_course_table(TABLE_TOOL_LIST);
  6034. $course_id = api_get_course_int_id();
  6035. $sql = "SELECT * FROM $t_tool
  6036. WHERE c_id = $course_id AND name = '".Database::escape_string($name)."' ";
  6037. $rs = Database::query($sql);
  6038. return Database::fetch_array($rs, 'ASSOC');
  6039. }
  6040. /**
  6041. * Function used to protect a "global" admin script.
  6042. * The function blocks access when the user has no global platform admin rights.
  6043. * Global admins are the admins that are registered in the main.admin table
  6044. * AND the users who have access to the "principal" portal.
  6045. * That means that there is a record in the main.access_url_rel_user table
  6046. * with his user id and the access_url_id=1.
  6047. *
  6048. * @author Julio Montoya
  6049. *
  6050. * @param int $user_id
  6051. *
  6052. * @return bool
  6053. */
  6054. function api_is_global_platform_admin($user_id = null)
  6055. {
  6056. $user_id = (int) $user_id;
  6057. if (empty($user_id)) {
  6058. $user_id = api_get_user_id();
  6059. }
  6060. if (api_is_platform_admin_by_id($user_id)) {
  6061. $urlList = api_get_access_url_from_user($user_id);
  6062. // The admin is registered in the first "main" site with access_url_id = 1
  6063. if (in_array(1, $urlList)) {
  6064. return true;
  6065. } else {
  6066. return false;
  6067. }
  6068. }
  6069. return false;
  6070. }
  6071. /**
  6072. * @param int $admin_id_to_check
  6073. * @param int $my_user_id
  6074. * @param bool $allow_session_admin
  6075. *
  6076. * @return bool
  6077. */
  6078. function api_global_admin_can_edit_admin(
  6079. $admin_id_to_check,
  6080. $my_user_id = null,
  6081. $allow_session_admin = false
  6082. ) {
  6083. if (empty($my_user_id)) {
  6084. $my_user_id = api_get_user_id();
  6085. }
  6086. $iam_a_global_admin = api_is_global_platform_admin($my_user_id);
  6087. $user_is_global_admin = api_is_global_platform_admin($admin_id_to_check);
  6088. if ($iam_a_global_admin) {
  6089. // Global admin can edit everything
  6090. return true;
  6091. } else {
  6092. // If i'm a simple admin
  6093. $is_platform_admin = api_is_platform_admin_by_id($my_user_id);
  6094. if ($allow_session_admin) {
  6095. $is_platform_admin = api_is_platform_admin_by_id($my_user_id) || (api_get_user_status($my_user_id) == SESSIONADMIN);
  6096. }
  6097. if ($is_platform_admin) {
  6098. if ($user_is_global_admin) {
  6099. return false;
  6100. } else {
  6101. return true;
  6102. }
  6103. } else {
  6104. return false;
  6105. }
  6106. }
  6107. }
  6108. /**
  6109. * @param int $admin_id_to_check
  6110. * @param int $my_user_id
  6111. * @param bool $allow_session_admin
  6112. *
  6113. * @return bool|null
  6114. */
  6115. function api_protect_super_admin($admin_id_to_check, $my_user_id = null, $allow_session_admin = false)
  6116. {
  6117. if (api_global_admin_can_edit_admin($admin_id_to_check, $my_user_id, $allow_session_admin)) {
  6118. return true;
  6119. } else {
  6120. api_not_allowed();
  6121. }
  6122. }
  6123. /**
  6124. * Function used to protect a global admin script.
  6125. * The function blocks access when the user has no global platform admin rights.
  6126. * See also the api_is_global_platform_admin() function wich defines who's a "global" admin.
  6127. *
  6128. * @author Julio Montoya
  6129. */
  6130. function api_protect_global_admin_script()
  6131. {
  6132. if (!api_is_global_platform_admin()) {
  6133. api_not_allowed();
  6134. return false;
  6135. }
  6136. return true;
  6137. }
  6138. /**
  6139. * Get active template.
  6140. *
  6141. * @param string theme type (optional: default)
  6142. * @param string path absolute(abs) or relative(rel) (optional:rel)
  6143. *
  6144. * @return string actived template path
  6145. */
  6146. function api_get_template($path_type = 'rel')
  6147. {
  6148. $path_types = ['rel', 'abs'];
  6149. $template_path = '';
  6150. if (in_array($path_type, $path_types)) {
  6151. if ($path_type == 'rel') {
  6152. $template_path = api_get_path(SYS_TEMPLATE_PATH);
  6153. } else {
  6154. $template_path = api_get_path(WEB_TEMPLATE_PATH);
  6155. }
  6156. }
  6157. $actived_theme = 'default';
  6158. if (api_get_setting('active_template')) {
  6159. $actived_theme = api_get_setting('active_template');
  6160. }
  6161. $actived_theme_path = $template_path.$actived_theme.DIRECTORY_SEPARATOR;
  6162. return $actived_theme_path;
  6163. }
  6164. /**
  6165. * Check browser support for specific file types or features
  6166. * This function checks if the user's browser supports a file format or given
  6167. * feature, or returns the current browser and major version when
  6168. * $format=check_browser. Only a limited number of formats and features are
  6169. * checked by this method. Make sure you check its definition first.
  6170. *
  6171. * @param string $format Can be a file format (extension like svg, webm, ...) or a feature (like autocapitalize, ...)
  6172. *
  6173. * @return bool or return text array if $format=check_browser
  6174. *
  6175. * @author Juan Carlos Raña Trabado
  6176. */
  6177. function api_browser_support($format = '')
  6178. {
  6179. $browser = new Browser();
  6180. $current_browser = $browser->getBrowser();
  6181. $a_versiontemp = explode('.', $browser->getVersion());
  6182. $current_majorver = $a_versiontemp[0];
  6183. static $result;
  6184. if (isset($result[$format])) {
  6185. return $result[$format];
  6186. }
  6187. // Native svg support
  6188. if ($format == 'svg') {
  6189. if (($current_browser == 'Internet Explorer' && $current_majorver >= 9) ||
  6190. ($current_browser == 'Firefox' && $current_majorver > 1) ||
  6191. ($current_browser == 'Safari' && $current_majorver >= 4) ||
  6192. ($current_browser == 'Chrome' && $current_majorver >= 1) ||
  6193. ($current_browser == 'Opera' && $current_majorver >= 9)
  6194. ) {
  6195. $result[$format] = true;
  6196. return true;
  6197. } else {
  6198. $result[$format] = false;
  6199. return false;
  6200. }
  6201. } elseif ($format == 'pdf') {
  6202. // native pdf support
  6203. if ($current_browser == 'Chrome' && $current_majorver >= 6) {
  6204. $result[$format] = true;
  6205. return true;
  6206. } else {
  6207. $result[$format] = false;
  6208. return false;
  6209. }
  6210. } elseif ($format == 'tif' || $format == 'tiff') {
  6211. //native tif support
  6212. if ($current_browser == 'Safari' && $current_majorver >= 5) {
  6213. $result[$format] = true;
  6214. return true;
  6215. } else {
  6216. $result[$format] = false;
  6217. return false;
  6218. }
  6219. } elseif ($format == 'ogg' || $format == 'ogx' || $format == 'ogv' || $format == 'oga') {
  6220. //native ogg, ogv,oga support
  6221. if (($current_browser == 'Firefox' && $current_majorver >= 3) ||
  6222. ($current_browser == 'Chrome' && $current_majorver >= 3) ||
  6223. ($current_browser == 'Opera' && $current_majorver >= 9)) {
  6224. $result[$format] = true;
  6225. return true;
  6226. } else {
  6227. $result[$format] = false;
  6228. return false;
  6229. }
  6230. } elseif ($format == 'mpg' || $format == 'mpeg') {
  6231. //native mpg support
  6232. if (($current_browser == 'Safari' && $current_majorver >= 5)) {
  6233. $result[$format] = true;
  6234. return true;
  6235. } else {
  6236. $result[$format] = false;
  6237. return false;
  6238. }
  6239. } elseif ($format == 'mp4') {
  6240. //native mp4 support (TODO: Android, iPhone)
  6241. if ($current_browser == 'Android' || $current_browser == 'iPhone') {
  6242. $result[$format] = true;
  6243. return true;
  6244. } else {
  6245. $result[$format] = false;
  6246. return false;
  6247. }
  6248. } elseif ($format == 'mov') {
  6249. //native mov support( TODO:check iPhone)
  6250. if ($current_browser == 'Safari' && $current_majorver >= 5 || $current_browser == 'iPhone') {
  6251. $result[$format] = true;
  6252. return true;
  6253. } else {
  6254. $result[$format] = false;
  6255. return false;
  6256. }
  6257. } elseif ($format == 'avi') {
  6258. //native avi support
  6259. if ($current_browser == 'Safari' && $current_majorver >= 5) {
  6260. $result[$format] = true;
  6261. return true;
  6262. } else {
  6263. $result[$format] = false;
  6264. return false;
  6265. }
  6266. } elseif ($format == 'wmv') {
  6267. //native wmv support
  6268. if ($current_browser == 'Firefox' && $current_majorver >= 4) {
  6269. $result[$format] = true;
  6270. return true;
  6271. } else {
  6272. $result[$format] = false;
  6273. return false;
  6274. }
  6275. } elseif ($format == 'webm') {
  6276. //native webm support (TODO:check IE9, Chrome9, Android)
  6277. if (($current_browser == 'Firefox' && $current_majorver >= 4) ||
  6278. ($current_browser == 'Opera' && $current_majorver >= 9) ||
  6279. ($current_browser == 'Internet Explorer' && $current_majorver >= 9) ||
  6280. ($current_browser == 'Chrome' && $current_majorver >= 9) ||
  6281. $current_browser == 'Android'
  6282. ) {
  6283. $result[$format] = true;
  6284. return true;
  6285. } else {
  6286. $result[$format] = false;
  6287. return false;
  6288. }
  6289. } elseif ($format == 'wav') {
  6290. //native wav support (only some codecs !)
  6291. if (($current_browser == 'Firefox' && $current_majorver >= 4) ||
  6292. ($current_browser == 'Safari' && $current_majorver >= 5) ||
  6293. ($current_browser == 'Opera' && $current_majorver >= 9) ||
  6294. ($current_browser == 'Internet Explorer' && $current_majorver >= 9) ||
  6295. ($current_browser == 'Chrome' && $current_majorver > 9) ||
  6296. $current_browser == 'Android' ||
  6297. $current_browser == 'iPhone'
  6298. ) {
  6299. $result[$format] = true;
  6300. return true;
  6301. } else {
  6302. $result[$format] = false;
  6303. return false;
  6304. }
  6305. } elseif ($format == 'mid' || $format == 'kar') {
  6306. //native midi support (TODO:check Android)
  6307. if ($current_browser == 'Opera' && $current_majorver >= 9 || $current_browser == 'Android') {
  6308. $result[$format] = true;
  6309. return true;
  6310. } else {
  6311. $result[$format] = false;
  6312. return false;
  6313. }
  6314. } elseif ($format == 'wma') {
  6315. //native wma support
  6316. if ($current_browser == 'Firefox' && $current_majorver >= 4) {
  6317. $result[$format] = true;
  6318. return true;
  6319. } else {
  6320. $result[$format] = false;
  6321. return false;
  6322. }
  6323. } elseif ($format == 'au') {
  6324. //native au support
  6325. if ($current_browser == 'Safari' && $current_majorver >= 5) {
  6326. $result[$format] = true;
  6327. return true;
  6328. } else {
  6329. $result[$format] = false;
  6330. return false;
  6331. }
  6332. } elseif ($format == 'mp3') {
  6333. //native mp3 support (TODO:check Android, iPhone)
  6334. if (($current_browser == 'Safari' && $current_majorver >= 5) ||
  6335. ($current_browser == 'Chrome' && $current_majorver >= 6) ||
  6336. ($current_browser == 'Internet Explorer' && $current_majorver >= 9) ||
  6337. $current_browser == 'Android' ||
  6338. $current_browser == 'iPhone' ||
  6339. $current_browser == 'Firefox'
  6340. ) {
  6341. $result[$format] = true;
  6342. return true;
  6343. } else {
  6344. $result[$format] = false;
  6345. return false;
  6346. }
  6347. } elseif ($format == 'autocapitalize') {
  6348. // Help avoiding showing the autocapitalize option if the browser doesn't
  6349. // support it: this attribute is against the HTML5 standard
  6350. if ($current_browser == 'Safari' || $current_browser == 'iPhone') {
  6351. return true;
  6352. } else {
  6353. return false;
  6354. }
  6355. } elseif ($format == "check_browser") {
  6356. $array_check_browser = [$current_browser, $current_majorver];
  6357. return $array_check_browser;
  6358. } else {
  6359. $result[$format] = false;
  6360. return false;
  6361. }
  6362. }
  6363. /**
  6364. * This function checks if exist path and file browscap.ini
  6365. * In order for this to work, your browscap configuration setting in php.ini
  6366. * must point to the correct location of the browscap.ini file on your system
  6367. * http://php.net/manual/en/function.get-browser.php.
  6368. *
  6369. * @return bool
  6370. *
  6371. * @author Juan Carlos Raña Trabado
  6372. */
  6373. function api_check_browscap()
  6374. {
  6375. $setting = ini_get('browscap');
  6376. if ($setting) {
  6377. $browser = get_browser($_SERVER['HTTP_USER_AGENT'], true);
  6378. if (strpos($setting, 'browscap.ini') && !empty($browser)) {
  6379. return true;
  6380. }
  6381. }
  6382. return false;
  6383. }
  6384. /**
  6385. * Returns the <script> HTML tag.
  6386. */
  6387. function api_get_js($file)
  6388. {
  6389. return '<script type="text/javascript" src="'.api_get_path(WEB_LIBRARY_PATH).'javascript/'.$file.'"></script>'."\n";
  6390. }
  6391. /**
  6392. * Returns the <script> HTML tag.
  6393. *
  6394. * @return string
  6395. */
  6396. function api_get_asset($file)
  6397. {
  6398. return '<script type="text/javascript" src="'.api_get_path(WEB_PUBLIC_PATH).'assets/'.$file.'"></script>'."\n";
  6399. }
  6400. /**
  6401. * Returns the <script> HTML tag.
  6402. *
  6403. * @param string $file
  6404. * @param string $media
  6405. *
  6406. * @return string
  6407. */
  6408. function api_get_css_asset($file, $media = 'screen')
  6409. {
  6410. return '<link href="'.api_get_path(WEB_PUBLIC_PATH).'assets/'.$file.'" rel="stylesheet" media="'.$media.'" type="text/css" />'."\n";
  6411. }
  6412. /**
  6413. * Returns the <link> HTML tag.
  6414. *
  6415. * @param string $file
  6416. * @param string $media
  6417. */
  6418. function api_get_css($file, $media = 'screen')
  6419. {
  6420. return '<link href="'.$file.'" rel="stylesheet" media="'.$media.'" type="text/css" />'."\n";
  6421. }
  6422. /**
  6423. * Returns the js header to include the jquery library.
  6424. */
  6425. function api_get_jquery_js()
  6426. {
  6427. return api_get_asset('jquery/dist/jquery.min.js');
  6428. }
  6429. /**
  6430. * Returns the jquery path.
  6431. *
  6432. * @return string
  6433. */
  6434. function api_get_jquery_web_path()
  6435. {
  6436. return api_get_path(WEB_PUBLIC_PATH).'assets/jquery/dist/jquery.min.js';
  6437. }
  6438. /**
  6439. * @return string
  6440. */
  6441. function api_get_jquery_ui_js_web_path()
  6442. {
  6443. return api_get_path(WEB_PUBLIC_PATH).'assets/jquery-ui/jquery-ui.min.js';
  6444. }
  6445. /**
  6446. * @return string
  6447. */
  6448. function api_get_jquery_ui_css_web_path()
  6449. {
  6450. return api_get_path(WEB_PUBLIC_PATH).'assets/jquery-ui/themes/smoothness/jquery-ui.min.css';
  6451. }
  6452. /**
  6453. * Returns the jquery-ui library js headers.
  6454. *
  6455. * @param bool add the jqgrid library
  6456. *
  6457. * @return string html tags
  6458. */
  6459. function api_get_jquery_ui_js($include_jqgrid = false)
  6460. {
  6461. $libraries = [];
  6462. if ($include_jqgrid) {
  6463. $libraries[] = 'jqgrid';
  6464. }
  6465. return api_get_jquery_libraries_js($libraries);
  6466. }
  6467. function api_get_jqgrid_js()
  6468. {
  6469. return api_get_jquery_libraries_js(['jqgrid']);
  6470. }
  6471. /**
  6472. * Returns the jquery library js and css headers.
  6473. *
  6474. * @param array list of jquery libraries supported jquery-ui, jqgrid
  6475. * @param bool add the jquery library
  6476. *
  6477. * @return string html tags
  6478. */
  6479. function api_get_jquery_libraries_js($libraries)
  6480. {
  6481. $js = '';
  6482. $js_path = api_get_path(WEB_LIBRARY_PATH).'javascript/';
  6483. //jqgrid js and css
  6484. if (in_array('jqgrid', $libraries)) {
  6485. $languaje = 'en';
  6486. $platform_isocode = strtolower(api_get_language_isocode());
  6487. //languages supported by jqgrid see files in main/inc/lib/javascript/jqgrid/js/i18n
  6488. $jqgrid_langs = [
  6489. 'bg', 'bg1251', 'cat', 'cn', 'cs', 'da', 'de', 'el', 'en', 'es', 'fa', 'fi', 'fr', 'gl', 'he', 'hu', 'is', 'it', 'ja', 'nl', 'no', 'pl', 'pt-br', 'pt', 'ro', 'ru', 'sk', 'sr', 'sv', 'tr', 'ua',
  6490. ];
  6491. if (in_array($platform_isocode, $jqgrid_langs)) {
  6492. $languaje = $platform_isocode;
  6493. }
  6494. //$js .= '<link rel="stylesheet" href="'.$js_path.'jqgrid/css/ui.jqgrid.css" type="text/css">';
  6495. $js .= api_get_css($js_path.'jqgrid/css/ui.jqgrid.css');
  6496. $js .= api_get_js('jqgrid/js/i18n/grid.locale-'.$languaje.'.js');
  6497. $js .= api_get_js('jqgrid/js/jquery.jqGrid.min.js');
  6498. }
  6499. //Document multiple upload funcionality
  6500. if (in_array('jquery-upload', $libraries)) {
  6501. $js .= api_get_asset('blueimp-load-image/js/load-image.all.min.js');
  6502. $js .= api_get_asset('blueimp-canvas-to-blob/js/canvas-to-blob.min.js');
  6503. $js .= api_get_asset('jquery-file-upload/js/jquery.iframe-transport.js');
  6504. $js .= api_get_asset('jquery-file-upload/js/jquery.fileupload.js');
  6505. $js .= api_get_asset('jquery-file-upload/js/jquery.fileupload-process.js');
  6506. $js .= api_get_asset('jquery-file-upload/js/jquery.fileupload-image.js');
  6507. $js .= api_get_asset('jquery-file-upload/js/jquery.fileupload-audio.js');
  6508. $js .= api_get_asset('jquery-file-upload/js/jquery.fileupload-video.js');
  6509. $js .= api_get_asset('jquery-file-upload/js/jquery.fileupload-validate.js');
  6510. $js .= api_get_css(api_get_path(WEB_PUBLIC_PATH).'assets/jquery-file-upload/css/jquery.fileupload.css');
  6511. $js .= api_get_css(api_get_path(WEB_PUBLIC_PATH).'assets/jquery-file-upload/css/jquery.fileupload-ui.css');
  6512. }
  6513. // jquery datepicker
  6514. if (in_array('datepicker', $libraries)) {
  6515. $languaje = 'en-GB';
  6516. $platform_isocode = strtolower(api_get_language_isocode());
  6517. // languages supported by jqgrid see files in main/inc/lib/javascript/jqgrid/js/i18n
  6518. $datapicker_langs = [
  6519. 'af', 'ar', 'ar-DZ', 'az', 'bg', 'bs', 'ca', 'cs', 'cy-GB', 'da', 'de', 'el', 'en-AU', 'en-GB', 'en-NZ', 'eo', 'es', 'et', 'eu', 'fa', 'fi', 'fo', 'fr', 'fr-CH', 'gl', 'he', 'hi', 'hr', 'hu', 'hy', 'id', 'is', 'it', 'ja', 'ka', 'kk', 'km', 'ko', 'lb', 'lt', 'lv', 'mk', 'ml', 'ms', 'nl', 'nl-BE', 'no', 'pl', 'pt', 'pt-BR', 'rm', 'ro', 'ru', 'sk', 'sl', 'sq', 'sr', 'sr-SR', 'sv', 'ta', 'th', 'tj', 'tr', 'uk', 'vi', 'zh-CN', 'zh-HK', 'zh-TW',
  6520. ];
  6521. if (in_array($platform_isocode, $datapicker_langs)) {
  6522. $languaje = $platform_isocode;
  6523. }
  6524. $js .= api_get_js('jquery-ui/jquery-ui-i18n.min.js');
  6525. $script = '<script>
  6526. $(function(){
  6527. $.datepicker.setDefaults($.datepicker.regional["'.$languaje.'"]);
  6528. $.datepicker.regional["local"] = $.datepicker.regional["'.$languaje.'"];
  6529. });
  6530. </script>
  6531. ';
  6532. $js .= $script;
  6533. }
  6534. return $js;
  6535. }
  6536. /**
  6537. * Returns the URL to the course or session, removing the complexity of the URL
  6538. * building piece by piece.
  6539. *
  6540. * This function relies on api_get_course_info()
  6541. *
  6542. * @param string $courseCode The course code - optional (takes it from context if not given)
  6543. * @param int $sessionId The session ID - optional (takes it from context if not given)
  6544. * @param int $groupId The group ID - optional (takes it from context if not given)
  6545. *
  6546. * @return string The URL to a course, a session, or empty string if nothing works e.g. https://localhost/courses/ABC/index.php?session_id=3&gidReq=1
  6547. *
  6548. * @author Julio Montoya <gugli100@gmail.com>
  6549. */
  6550. function api_get_course_url($courseCode = null, $sessionId = null, $groupId = null)
  6551. {
  6552. $courseDirectory = '';
  6553. $url = '';
  6554. // If courseCode not set, get context or []
  6555. if (empty($courseCode)) {
  6556. $courseInfo = api_get_course_info();
  6557. } else {
  6558. $courseInfo = api_get_course_info($courseCode);
  6559. }
  6560. // If course defined, get directory, otherwise keep empty string
  6561. if (!empty($courseInfo['directory'])) {
  6562. $courseDirectory = $courseInfo['directory'];
  6563. }
  6564. // If sessionId not set, get context or 0
  6565. if (empty($sessionId)) {
  6566. $sessionId = api_get_session_id();
  6567. }
  6568. // If groupId not set, get context or 0
  6569. if (empty($groupId)) {
  6570. $groupId = api_get_group_id();
  6571. }
  6572. // Build the URL
  6573. if (!empty($courseDirectory)) {
  6574. // directory not empty, so we do have a course
  6575. $url = api_get_path(WEB_COURSE_PATH).$courseDirectory.'/index.php?id_session='.$sessionId.'&gidReq='.$groupId;
  6576. } elseif (!empty($sessionId) && api_get_configuration_value('remove_session_url') !== true) {
  6577. // if the course was unset and the session was set, send directly to the session
  6578. $url = api_get_path(WEB_CODE_PATH).'session/index.php?session_id='.$sessionId;
  6579. }
  6580. // if not valid combination was found, return an empty string
  6581. return $url;
  6582. }
  6583. /**
  6584. * Check if the current portal has the $_configuration['multiple_access_urls'] parameter on.
  6585. *
  6586. * @return bool true if multi site is enabled
  6587. */
  6588. function api_get_multiple_access_url()
  6589. {
  6590. global $_configuration;
  6591. if (isset($_configuration['multiple_access_urls']) && $_configuration['multiple_access_urls']) {
  6592. return true;
  6593. }
  6594. return false;
  6595. }
  6596. /**
  6597. * @return bool
  6598. */
  6599. function api_is_multiple_url_enabled()
  6600. {
  6601. return api_get_multiple_access_url();
  6602. }
  6603. /**
  6604. * Returns a md5 unique id.
  6605. *
  6606. * @todo add more parameters
  6607. */
  6608. function api_get_unique_id()
  6609. {
  6610. $id = md5(time().uniqid().api_get_user_id().api_get_course_id().api_get_session_id());
  6611. return $id;
  6612. }
  6613. /**
  6614. * Get home path.
  6615. *
  6616. * @return string
  6617. */
  6618. function api_get_home_path()
  6619. {
  6620. // FIX : Start the routing determination from central path definition
  6621. $home = api_get_path(SYS_HOME_PATH);
  6622. if (api_get_multiple_access_url()) {
  6623. $access_url_id = api_get_current_access_url_id();
  6624. $url_info = api_get_access_url($access_url_id);
  6625. $url = api_remove_trailing_slash(preg_replace('/https?:\/\//i', '', $url_info['url']));
  6626. $clean_url = api_replace_dangerous_char($url);
  6627. $clean_url = str_replace('/', '-', $clean_url);
  6628. $clean_url .= '/';
  6629. if ($clean_url != 'localhost/') {
  6630. // means that the multiple URL was not well configured we don't rename the $home variable
  6631. return "{$home}{$clean_url}";
  6632. }
  6633. }
  6634. return $home;
  6635. }
  6636. /**
  6637. * @param int Course id
  6638. * @param int tool id: TOOL_QUIZ, TOOL_FORUM, TOOL_STUDENTPUBLICATION, TOOL_LEARNPATH
  6639. * @param int the item id (tool id, exercise id, lp id)
  6640. *
  6641. * @return bool
  6642. */
  6643. function api_resource_is_locked_by_gradebook($item_id, $link_type, $course_code = null)
  6644. {
  6645. if (api_is_platform_admin()) {
  6646. return false;
  6647. }
  6648. if (api_get_setting('gradebook_locking_enabled') == 'true') {
  6649. if (empty($course_code)) {
  6650. $course_code = api_get_course_id();
  6651. }
  6652. $table = Database::get_main_table(TABLE_MAIN_GRADEBOOK_LINK);
  6653. $item_id = intval($item_id);
  6654. $link_type = intval($link_type);
  6655. $course_code = Database::escape_string($course_code);
  6656. $sql = "SELECT locked FROM $table
  6657. WHERE locked = 1 AND ref_id = $item_id AND type = $link_type AND course_code = '$course_code' ";
  6658. $result = Database::query($sql);
  6659. if (Database::num_rows($result)) {
  6660. return true;
  6661. }
  6662. }
  6663. return false;
  6664. }
  6665. /**
  6666. * Blocks a page if the item was added in a gradebook.
  6667. *
  6668. * @param int exercise id, work id, thread id,
  6669. * @param int LINK_EXERCISE, LINK_STUDENTPUBLICATION, LINK_LEARNPATH LINK_FORUM_THREAD, LINK_ATTENDANCE
  6670. * see gradebook/lib/be/linkfactory
  6671. * @param string course code
  6672. *
  6673. * @return false|null
  6674. */
  6675. function api_block_course_item_locked_by_gradebook($item_id, $link_type, $course_code = null)
  6676. {
  6677. if (api_is_platform_admin()) {
  6678. return false;
  6679. }
  6680. if (api_resource_is_locked_by_gradebook($item_id, $link_type, $course_code)) {
  6681. $message = Display::return_message(get_lang('ResourceLockedByGradebook'), 'warning');
  6682. api_not_allowed(true, $message);
  6683. }
  6684. }
  6685. /**
  6686. * Checks the PHP version installed is enough to run Chamilo.
  6687. *
  6688. * @param string Include path (used to load the error page)
  6689. */
  6690. function api_check_php_version($my_inc_path = null)
  6691. {
  6692. if (!function_exists('version_compare') || version_compare(phpversion(), REQUIRED_PHP_VERSION, '<')) {
  6693. $global_error_code = 1;
  6694. // Incorrect PHP version
  6695. $global_page = $my_inc_path.'global_error_message.inc.php';
  6696. if (file_exists($global_page)) {
  6697. require $global_page;
  6698. }
  6699. exit;
  6700. }
  6701. }
  6702. /**
  6703. * Checks whether the Archive directory is present and writeable. If not,
  6704. * prints a warning message.
  6705. */
  6706. function api_check_archive_dir()
  6707. {
  6708. if (is_dir(api_get_path(SYS_ARCHIVE_PATH)) && !is_writable(api_get_path(SYS_ARCHIVE_PATH))) {
  6709. $message = Display::return_message(get_lang('ArchivesDirectoryNotWriteableContactAdmin'), 'warning');
  6710. api_not_allowed(true, $message);
  6711. }
  6712. }
  6713. /**
  6714. * Returns an array of global configuration settings which should be ignored
  6715. * when printing the configuration settings screens.
  6716. *
  6717. * @return array Array of strings, each identifying one of the excluded settings
  6718. */
  6719. function api_get_locked_settings()
  6720. {
  6721. return [
  6722. 'server_type',
  6723. 'permanently_remove_deleted_files',
  6724. 'account_valid_duration',
  6725. 'service_ppt2lp',
  6726. 'wcag_anysurfer_public_pages',
  6727. 'upload_extensions_list_type',
  6728. 'upload_extensions_blacklist',
  6729. 'upload_extensions_whitelist',
  6730. 'upload_extensions_skip',
  6731. 'upload_extensions_replace_by',
  6732. 'hide_dltt_markup',
  6733. 'split_users_upload_directory',
  6734. 'permissions_for_new_directories',
  6735. 'permissions_for_new_files',
  6736. 'platform_charset',
  6737. 'ldap_description',
  6738. 'cas_activate',
  6739. 'cas_server',
  6740. 'cas_server_uri',
  6741. 'cas_port',
  6742. 'cas_protocol',
  6743. 'cas_add_user_activate',
  6744. 'update_user_info_cas_with_ldap',
  6745. 'languagePriority1',
  6746. 'languagePriority2',
  6747. 'languagePriority3',
  6748. 'languagePriority4',
  6749. 'login_is_email',
  6750. 'chamilo_database_version',
  6751. ];
  6752. }
  6753. /**
  6754. * Checks if the user is corrently logged in. Returns the user ID if he is, or
  6755. * false if he isn't. If the user ID is given and is an integer, then the same
  6756. * ID is simply returned.
  6757. *
  6758. * @param int User ID
  6759. *
  6760. * @return bool Integer User ID is logged in, or false otherwise
  6761. */
  6762. function api_user_is_login($user_id = null)
  6763. {
  6764. $user_id = empty($user_id) ? api_get_user_id() : (int) $user_id;
  6765. return $user_id && !api_is_anonymous();
  6766. }
  6767. /**
  6768. * Guess the real ip for register in the database, even in reverse proxy cases.
  6769. * To be recognized, the IP has to be found in either $_SERVER['REMOTE_ADDR'] or
  6770. * in $_SERVER['HTTP_X_FORWARDED_FOR'], which is in common use with rproxies.
  6771. * Note: the result of this function is not SQL-safe. Please escape it before
  6772. * inserting in a database.
  6773. *
  6774. * @return string the user's real ip (unsafe - escape it before inserting to db)
  6775. *
  6776. * @author Jorge Frisancho Jibaja <jrfdeft@gmail.com>, USIL - Some changes to allow the use of real IP using reverse proxy
  6777. *
  6778. * @version CEV CHANGE 24APR2012
  6779. */
  6780. function api_get_real_ip()
  6781. {
  6782. $ip = trim($_SERVER['REMOTE_ADDR']);
  6783. if (!empty($_SERVER['HTTP_X_FORWARDED_FOR'])) {
  6784. if (preg_match('/,/', $_SERVER['HTTP_X_FORWARDED_FOR'])) {
  6785. @list($ip1, $ip2) = @explode(',', $_SERVER['HTTP_X_FORWARDED_FOR']);
  6786. } else {
  6787. $ip1 = $_SERVER['HTTP_X_FORWARDED_FOR'];
  6788. }
  6789. $ip = trim($ip1);
  6790. }
  6791. return $ip;
  6792. }
  6793. /**
  6794. * Checks whether an IP is included inside an IP range.
  6795. *
  6796. * @param string IP address
  6797. * @param string IP range
  6798. * @param string $ip
  6799. *
  6800. * @return bool True if IP is in the range, false otherwise
  6801. *
  6802. * @author claudiu at cnixs dot com on http://www.php.net/manual/fr/ref.network.php#55230
  6803. * @author Yannick Warnier for improvements and managment of multiple ranges
  6804. *
  6805. * @todo check for IPv6 support
  6806. */
  6807. function api_check_ip_in_range($ip, $range)
  6808. {
  6809. if (empty($ip) or empty($range)) {
  6810. return false;
  6811. }
  6812. $ip_ip = ip2long($ip);
  6813. // divide range param into array of elements
  6814. if (strpos($range, ',') !== false) {
  6815. $ranges = explode(',', $range);
  6816. } else {
  6817. $ranges = [$range];
  6818. }
  6819. foreach ($ranges as $range) {
  6820. $range = trim($range);
  6821. if (empty($range)) {
  6822. continue;
  6823. }
  6824. if (strpos($range, '/') === false) {
  6825. if (strcmp($ip, $range) === 0) {
  6826. return true; // there is a direct IP match, return OK
  6827. }
  6828. continue; //otherwise, get to the next range
  6829. }
  6830. // the range contains a "/", so analyse completely
  6831. list($net, $mask) = explode("/", $range);
  6832. $ip_net = ip2long($net);
  6833. // mask binary magic
  6834. $ip_mask = ~((1 << (32 - $mask)) - 1);
  6835. $ip_ip_net = $ip_ip & $ip_mask;
  6836. if ($ip_ip_net == $ip_net) {
  6837. return true;
  6838. }
  6839. }
  6840. return false;
  6841. }
  6842. function api_check_user_access_to_legal($course_visibility)
  6843. {
  6844. $course_visibility_list = [COURSE_VISIBILITY_OPEN_WORLD, COURSE_VISIBILITY_OPEN_PLATFORM];
  6845. return in_array($course_visibility, $course_visibility_list) || api_is_drh();
  6846. }
  6847. /**
  6848. * Checks if the global chat is enabled or not.
  6849. *
  6850. * @return bool
  6851. */
  6852. function api_is_global_chat_enabled()
  6853. {
  6854. return
  6855. !api_is_anonymous() &&
  6856. api_get_setting('allow_global_chat') === 'true' &&
  6857. api_get_setting('allow_social_tool') === 'true';
  6858. }
  6859. /**
  6860. * @todo Fix tool_visible_by_default_at_creation labels
  6861. *
  6862. * @param int $item_id
  6863. * @param int $tool_id
  6864. * @param int $group_id id
  6865. * @param array $courseInfo
  6866. * @param int $sessionId
  6867. * @param int $userId
  6868. */
  6869. function api_set_default_visibility(
  6870. $item_id,
  6871. $tool_id,
  6872. $group_id = 0,
  6873. $courseInfo = [],
  6874. $sessionId = 0,
  6875. $userId = 0
  6876. ) {
  6877. $courseInfo = empty($courseInfo) ? api_get_course_info() : $courseInfo;
  6878. $courseId = $courseInfo['real_id'];
  6879. $courseCode = $courseInfo['code'];
  6880. $sessionId = empty($sessionId) ? api_get_session_id() : $sessionId;
  6881. $userId = empty($userId) ? api_get_user_id() : $userId;
  6882. // if group is null force group_id = 0, this force is needed to create a LP folder with group = 0
  6883. if (is_null($group_id)) {
  6884. $group_id = 0;
  6885. } else {
  6886. $group_id = empty($group_id) ? api_get_group_id() : $group_id;
  6887. }
  6888. $groupInfo = [];
  6889. if (!empty($group_id)) {
  6890. $groupInfo = GroupManager::get_group_properties($group_id);
  6891. }
  6892. $original_tool_id = $tool_id;
  6893. switch ($tool_id) {
  6894. case TOOL_LINK:
  6895. case TOOL_LINK_CATEGORY:
  6896. $tool_id = 'links';
  6897. break;
  6898. case TOOL_DOCUMENT:
  6899. $tool_id = 'documents';
  6900. break;
  6901. case TOOL_LEARNPATH:
  6902. $tool_id = 'learning';
  6903. break;
  6904. case TOOL_ANNOUNCEMENT:
  6905. $tool_id = 'announcements';
  6906. break;
  6907. case TOOL_FORUM:
  6908. case TOOL_FORUM_CATEGORY:
  6909. case TOOL_FORUM_THREAD:
  6910. $tool_id = 'forums';
  6911. break;
  6912. case TOOL_QUIZ:
  6913. $tool_id = 'quiz';
  6914. break;
  6915. }
  6916. $setting = api_get_setting('tool_visible_by_default_at_creation');
  6917. if (isset($setting[$tool_id])) {
  6918. $visibility = 'invisible';
  6919. if ($setting[$tool_id] == 'true') {
  6920. $visibility = 'visible';
  6921. }
  6922. // Read the portal and course default visibility
  6923. if ($tool_id == 'documents') {
  6924. $visibility = DocumentManager::getDocumentDefaultVisibility($courseCode);
  6925. }
  6926. api_item_property_update(
  6927. $courseInfo,
  6928. $original_tool_id,
  6929. $item_id,
  6930. $visibility,
  6931. $userId,
  6932. $groupInfo,
  6933. null,
  6934. null,
  6935. null,
  6936. $sessionId
  6937. );
  6938. // Fixes default visibility for tests
  6939. switch ($original_tool_id) {
  6940. case TOOL_QUIZ:
  6941. if (empty($sessionId)) {
  6942. $objExerciseTmp = new Exercise($courseId);
  6943. $objExerciseTmp->read($item_id);
  6944. if ($visibility == 'visible') {
  6945. $objExerciseTmp->enable();
  6946. $objExerciseTmp->save();
  6947. } else {
  6948. $objExerciseTmp->disable();
  6949. $objExerciseTmp->save();
  6950. }
  6951. }
  6952. break;
  6953. }
  6954. }
  6955. }
  6956. /**
  6957. * @return string
  6958. */
  6959. function api_get_security_key()
  6960. {
  6961. return api_get_configuration_value('security_key');
  6962. }
  6963. /**
  6964. * @param int $user_id
  6965. * @param int $courseId
  6966. * @param int $session_id
  6967. *
  6968. * @return array
  6969. */
  6970. function api_detect_user_roles($user_id, $courseId, $session_id = 0)
  6971. {
  6972. $user_roles = [];
  6973. $courseInfo = api_get_course_info_by_id($courseId);
  6974. $course_code = $courseInfo['code'];
  6975. $url_id = api_get_current_access_url_id();
  6976. if (api_is_platform_admin_by_id($user_id, $url_id)) {
  6977. $user_roles[] = PLATFORM_ADMIN;
  6978. }
  6979. /*if (api_is_drh()) {
  6980. $user_roles[] = DRH;
  6981. }*/
  6982. if (!empty($session_id)) {
  6983. if (SessionManager::user_is_general_coach($user_id, $session_id)) {
  6984. $user_roles[] = SESSION_GENERAL_COACH;
  6985. }
  6986. }
  6987. if (!empty($course_code)) {
  6988. if (empty($session_id)) {
  6989. if (CourseManager::is_course_teacher($user_id, $course_code)) {
  6990. $user_roles[] = COURSEMANAGER;
  6991. }
  6992. if (CourseManager::get_tutor_in_course_status($user_id, $courseInfo['real_id'])) {
  6993. $user_roles[] = COURSE_TUTOR;
  6994. }
  6995. if (CourseManager::is_user_subscribed_in_course($user_id, $course_code)) {
  6996. $user_roles[] = COURSE_STUDENT;
  6997. }
  6998. } else {
  6999. $user_status_in_session = SessionManager::get_user_status_in_course_session(
  7000. $user_id,
  7001. $courseId,
  7002. $session_id
  7003. );
  7004. if (!empty($user_status_in_session)) {
  7005. if ($user_status_in_session == 0) {
  7006. $user_roles[] = SESSION_STUDENT;
  7007. }
  7008. if ($user_status_in_session == 2) {
  7009. $user_roles[] = SESSION_COURSE_COACH;
  7010. }
  7011. }
  7012. /*if (api_is_course_session_coach($user_id, $course_code, $session_id)) {
  7013. $user_roles[] = SESSION_COURSE_COACH;
  7014. }*/
  7015. }
  7016. }
  7017. return $user_roles;
  7018. }
  7019. /**
  7020. * @param int $courseId
  7021. * @param int $session_id
  7022. *
  7023. * @return bool
  7024. */
  7025. function api_coach_can_edit_view_results($courseId = null, $session_id = null)
  7026. {
  7027. if (api_is_platform_admin()) {
  7028. return true;
  7029. }
  7030. $user_id = api_get_user_id();
  7031. if (empty($courseId)) {
  7032. $courseId = api_get_course_int_id();
  7033. }
  7034. if (empty($session_id)) {
  7035. $session_id = api_get_session_id();
  7036. }
  7037. $roles = api_detect_user_roles($user_id, $courseId, $session_id);
  7038. if (in_array(SESSION_COURSE_COACH, $roles)) {
  7039. //return api_get_setting('session_tutor_reports_visibility') == 'true';
  7040. return true;
  7041. } else {
  7042. if (in_array(COURSEMANAGER, $roles)) {
  7043. return true;
  7044. }
  7045. return false;
  7046. }
  7047. }
  7048. /**
  7049. * @param string $file
  7050. *
  7051. * @return string
  7052. */
  7053. function api_get_js_simple($file)
  7054. {
  7055. return '<script type="text/javascript" src="'.$file.'"></script>'."\n";
  7056. }
  7057. function api_set_settings_and_plugins()
  7058. {
  7059. global $_configuration;
  7060. $_setting = [];
  7061. $_plugins = [];
  7062. // access_url == 1 is the default chamilo location
  7063. $settings_by_access_list = [];
  7064. $access_url_id = api_get_current_access_url_id();
  7065. if ($access_url_id != 1) {
  7066. $url_info = api_get_access_url($_configuration['access_url']);
  7067. if ($url_info['active'] == 1) {
  7068. $settings_by_access = &api_get_settings(null, 'list', $_configuration['access_url'], 1);
  7069. foreach ($settings_by_access as &$row) {
  7070. if (empty($row['variable'])) {
  7071. $row['variable'] = 0;
  7072. }
  7073. if (empty($row['subkey'])) {
  7074. $row['subkey'] = 0;
  7075. }
  7076. if (empty($row['category'])) {
  7077. $row['category'] = 0;
  7078. }
  7079. $settings_by_access_list[$row['variable']][$row['subkey']][$row['category']] = $row;
  7080. }
  7081. }
  7082. }
  7083. $result = api_get_settings(null, 'list', 1);
  7084. foreach ($result as &$row) {
  7085. if ($access_url_id != 1) {
  7086. if ($url_info['active'] == 1) {
  7087. $var = empty($row['variable']) ? 0 : $row['variable'];
  7088. $subkey = empty($row['subkey']) ? 0 : $row['subkey'];
  7089. $category = empty($row['category']) ? 0 : $row['category'];
  7090. }
  7091. if ($row['access_url_changeable'] == 1 && $url_info['active'] == 1) {
  7092. if (isset($settings_by_access_list[$var]) &&
  7093. $settings_by_access_list[$var][$subkey][$category]['selected_value'] != '') {
  7094. if ($row['subkey'] == null) {
  7095. $_setting[$row['variable']] = $settings_by_access_list[$var][$subkey][$category]['selected_value'];
  7096. } else {
  7097. $_setting[$row['variable']][$row['subkey']] = $settings_by_access_list[$var][$subkey][$category]['selected_value'];
  7098. }
  7099. } else {
  7100. if ($row['subkey'] == null) {
  7101. $_setting[$row['variable']] = $row['selected_value'];
  7102. } else {
  7103. $_setting[$row['variable']][$row['subkey']] = $row['selected_value'];
  7104. }
  7105. }
  7106. } else {
  7107. if ($row['subkey'] == null) {
  7108. $_setting[$row['variable']] = $row['selected_value'];
  7109. } else {
  7110. $_setting[$row['variable']][$row['subkey']] = $row['selected_value'];
  7111. }
  7112. }
  7113. } else {
  7114. if ($row['subkey'] == null) {
  7115. $_setting[$row['variable']] = $row['selected_value'];
  7116. } else {
  7117. $_setting[$row['variable']][$row['subkey']] = $row['selected_value'];
  7118. }
  7119. }
  7120. }
  7121. $result = api_get_settings('Plugins', 'list', $access_url_id);
  7122. $_plugins = [];
  7123. foreach ($result as &$row) {
  7124. $key = &$row['variable'];
  7125. if (is_string($_setting[$key])) {
  7126. $_setting[$key] = [];
  7127. }
  7128. $_setting[$key][] = $row['selected_value'];
  7129. $_plugins[$key][] = $row['selected_value'];
  7130. }
  7131. $_SESSION['_setting'] = $_setting;
  7132. $_SESSION['_plugins'] = $_plugins;
  7133. }
  7134. /**
  7135. * Modify default memory_limit and max_execution_time limits
  7136. * Needed when processing long tasks.
  7137. */
  7138. function api_set_more_memory_and_time_limits()
  7139. {
  7140. if (function_exists('ini_set')) {
  7141. api_set_memory_limit('256M');
  7142. ini_set('max_execution_time', 1800);
  7143. }
  7144. }
  7145. /**
  7146. * Tries to set memory limit, if authorized and new limit is higher than current.
  7147. *
  7148. * @param string $mem New memory limit
  7149. *
  7150. * @return bool True on success, false on failure or current is higher than suggested
  7151. * @assert (null) === false
  7152. * @assert (-1) === false
  7153. * @assert (0) === true
  7154. * @assert ('1G') === true
  7155. */
  7156. function api_set_memory_limit($mem)
  7157. {
  7158. //if ini_set() not available, this function is useless
  7159. if (!function_exists('ini_set') || is_null($mem) || $mem == -1) {
  7160. return false;
  7161. }
  7162. $memory_limit = ini_get('memory_limit');
  7163. if (api_get_bytes_memory_limit($mem) > api_get_bytes_memory_limit($memory_limit)) {
  7164. ini_set('memory_limit', $mem);
  7165. return true;
  7166. }
  7167. return false;
  7168. }
  7169. /**
  7170. * Gets memory limit in bytes.
  7171. *
  7172. * @param string The memory size (128M, 1G, 1000K, etc)
  7173. *
  7174. * @return int
  7175. * @assert (null) === false
  7176. * @assert ('1t') === 1099511627776
  7177. * @assert ('1g') === 1073741824
  7178. * @assert ('1m') === 1048576
  7179. * @assert ('100k') === 102400
  7180. */
  7181. function api_get_bytes_memory_limit($mem)
  7182. {
  7183. $size = strtolower(substr($mem, -1));
  7184. switch ($size) {
  7185. case 't':
  7186. $mem = intval(substr($mem, -1)) * 1024 * 1024 * 1024 * 1024;
  7187. break;
  7188. case 'g':
  7189. $mem = intval(substr($mem, 0, -1)) * 1024 * 1024 * 1024;
  7190. break;
  7191. case 'm':
  7192. $mem = intval(substr($mem, 0, -1)) * 1024 * 1024;
  7193. break;
  7194. case 'k':
  7195. $mem = intval(substr($mem, 0, -1)) * 1024;
  7196. break;
  7197. default:
  7198. // we assume it's integer only
  7199. $mem = intval($mem);
  7200. break;
  7201. }
  7202. return $mem;
  7203. }
  7204. /**
  7205. * Finds all the information about a user from username instead of user id.
  7206. *
  7207. * @param string $officialCode
  7208. *
  7209. * @return array $user_info user_id, lastname, firstname, username, email, ...
  7210. *
  7211. * @author Yannick Warnier <yannick.warnier@beeznest.com>
  7212. */
  7213. function api_get_user_info_from_official_code($officialCode)
  7214. {
  7215. if (empty($officialCode)) {
  7216. return false;
  7217. }
  7218. $sql = "SELECT * FROM ".Database::get_main_table(TABLE_MAIN_USER)."
  7219. WHERE official_code ='".Database::escape_string($officialCode)."'";
  7220. $result = Database::query($sql);
  7221. if (Database::num_rows($result) > 0) {
  7222. $result_array = Database::fetch_array($result);
  7223. return _api_format_user($result_array);
  7224. }
  7225. return false;
  7226. }
  7227. /**
  7228. * @param string $usernameInputId
  7229. * @param string $passwordInputId
  7230. *
  7231. * @return string|null
  7232. */
  7233. function api_get_password_checker_js($usernameInputId, $passwordInputId)
  7234. {
  7235. $checkPass = api_get_setting('allow_strength_pass_checker');
  7236. $useStrengthPassChecker = $checkPass === 'true';
  7237. if ($useStrengthPassChecker === false) {
  7238. return null;
  7239. }
  7240. $translations = [
  7241. 'wordLength' => get_lang('PasswordIsTooShort'),
  7242. 'wordNotEmail' => get_lang('YourPasswordCannotBeTheSameAsYourEmail'),
  7243. 'wordSimilarToUsername' => get_lang('YourPasswordCannotContainYourUsername'),
  7244. 'wordTwoCharacterClasses' => get_lang('WordTwoCharacterClasses'),
  7245. 'wordRepetitions' => get_lang('TooManyRepetitions'),
  7246. 'wordSequences' => get_lang('YourPasswordContainsSequences'),
  7247. 'errorList' => get_lang('ErrorsFound'),
  7248. 'veryWeak' => get_lang('PasswordVeryWeak'),
  7249. 'weak' => get_lang('PasswordWeak'),
  7250. 'normal' => get_lang('PasswordNormal'),
  7251. 'medium' => get_lang('PasswordMedium'),
  7252. 'strong' => get_lang('PasswordStrong'),
  7253. 'veryStrong' => get_lang('PasswordVeryStrong'),
  7254. ];
  7255. $js = api_get_asset('pwstrength-bootstrap/dist/pwstrength-bootstrap.min.js');
  7256. $js .= "<script>
  7257. var errorMessages = {
  7258. password_to_short : \"".get_lang('PasswordIsTooShort')."\",
  7259. same_as_username : \"".get_lang('YourPasswordCannotBeTheSameAsYourUsername')."\"
  7260. };
  7261. $(function() {
  7262. var lang = ".json_encode($translations).";
  7263. var options = {
  7264. onLoad : function () {
  7265. //$('#messages').text('Start typing password');
  7266. },
  7267. onKeyUp: function (evt) {
  7268. $(evt.target).pwstrength('outputErrorList');
  7269. },
  7270. errorMessages : errorMessages,
  7271. viewports: {
  7272. progress: '#password_progress',
  7273. verdict: '#password-verdict',
  7274. errors: '#password-errors'
  7275. },
  7276. usernameField: '$usernameInputId'
  7277. };
  7278. options.i18n = {
  7279. t: function (key) {
  7280. var result = lang[key];
  7281. return result === key ? '' : result; // This assumes you return the
  7282. }
  7283. };
  7284. $('".$passwordInputId."').pwstrength(options);
  7285. });
  7286. </script>";
  7287. return $js;
  7288. }
  7289. /**
  7290. * create an user extra field called 'captcha_blocked_until_date'.
  7291. *
  7292. * @param string $username
  7293. *
  7294. * @return bool
  7295. */
  7296. function api_block_account_captcha($username)
  7297. {
  7298. $userInfo = api_get_user_info_from_username($username);
  7299. if (empty($userInfo)) {
  7300. return false;
  7301. }
  7302. $minutesToBlock = api_get_setting('captcha_time_to_block');
  7303. $time = time() + $minutesToBlock * 60;
  7304. UserManager::update_extra_field_value(
  7305. $userInfo['user_id'],
  7306. 'captcha_blocked_until_date',
  7307. api_get_utc_datetime($time)
  7308. );
  7309. return true;
  7310. }
  7311. /**
  7312. * @param string $username
  7313. *
  7314. * @return bool
  7315. */
  7316. function api_clean_account_captcha($username)
  7317. {
  7318. $userInfo = api_get_user_info_from_username($username);
  7319. if (empty($userInfo)) {
  7320. return false;
  7321. }
  7322. Session::erase('loginFailedCount');
  7323. UserManager::update_extra_field_value(
  7324. $userInfo['user_id'],
  7325. 'captcha_blocked_until_date',
  7326. null
  7327. );
  7328. return true;
  7329. }
  7330. /**
  7331. * @param string $username
  7332. *
  7333. * @return bool
  7334. */
  7335. function api_get_user_blocked_by_captcha($username)
  7336. {
  7337. $userInfo = api_get_user_info_from_username($username);
  7338. if (empty($userInfo)) {
  7339. return false;
  7340. }
  7341. $data = UserManager::get_extra_user_data_by_field(
  7342. $userInfo['user_id'],
  7343. 'captcha_blocked_until_date'
  7344. );
  7345. if (isset($data) && isset($data['captcha_blocked_until_date'])) {
  7346. return $data['captcha_blocked_until_date'];
  7347. }
  7348. return false;
  7349. }
  7350. /**
  7351. * Remove tags from HTML anf return the $in_number_char first non-HTML char
  7352. * Postfix the text with "..." if it has been truncated.
  7353. *
  7354. * @param string $text
  7355. * @param int $number
  7356. *
  7357. * @return string
  7358. *
  7359. * @author hubert borderiou
  7360. */
  7361. function api_get_short_text_from_html($text, $number)
  7362. {
  7363. // Delete script and style tags
  7364. $text = preg_replace('/(<(script|style)\b[^>]*>).*?(<\/\2>)/is', "$1$3", $text);
  7365. $text = api_html_entity_decode($text);
  7366. $out_res = api_remove_tags_with_space($text, false);
  7367. $postfix = "...";
  7368. if (strlen($out_res) > $number) {
  7369. $out_res = substr($out_res, 0, $number).$postfix;
  7370. }
  7371. return $out_res;
  7372. }
  7373. /**
  7374. * Replace tags with a space in a text.
  7375. * If $in_double_quote_replace, replace " with '' (for HTML attribute purpose, for exemple).
  7376. *
  7377. * @return string
  7378. *
  7379. * @author hubert borderiou
  7380. */
  7381. function api_remove_tags_with_space($in_html, $in_double_quote_replace = true)
  7382. {
  7383. $out_res = $in_html;
  7384. if ($in_double_quote_replace) {
  7385. $out_res = str_replace('"', "''", $out_res);
  7386. }
  7387. // avoid text stuck together when tags are removed, adding a space after >
  7388. $out_res = str_replace(">", "> ", $out_res);
  7389. $out_res = strip_tags($out_res);
  7390. return $out_res;
  7391. }
  7392. /**
  7393. * If true, the drh can access all content (courses, users) inside a session.
  7394. *
  7395. * @return bool
  7396. */
  7397. function api_drh_can_access_all_session_content()
  7398. {
  7399. $value = api_get_setting('drh_can_access_all_session_content');
  7400. return $value === 'true';
  7401. }
  7402. /**
  7403. * @param string $tool
  7404. * @param string $setting
  7405. * @param int $defaultValue
  7406. *
  7407. * @return string
  7408. */
  7409. function api_get_default_tool_setting($tool, $setting, $defaultValue)
  7410. {
  7411. global $_configuration;
  7412. if (isset($_configuration[$tool]) &&
  7413. isset($_configuration[$tool]['default_settings']) &&
  7414. isset($_configuration[$tool]['default_settings'][$setting])
  7415. ) {
  7416. return $_configuration[$tool]['default_settings'][$setting];
  7417. }
  7418. return $defaultValue;
  7419. }
  7420. /**
  7421. * Checks if user can login as another user.
  7422. *
  7423. * @param int $loginAsUserId the user id to log in
  7424. * @param int $userId my user id
  7425. *
  7426. * @return bool
  7427. */
  7428. function api_can_login_as($loginAsUserId, $userId = null)
  7429. {
  7430. if (empty($userId)) {
  7431. $userId = api_get_user_id();
  7432. }
  7433. if ($loginAsUserId == $userId) {
  7434. return false;
  7435. }
  7436. if (empty($loginAsUserId)) {
  7437. return false;
  7438. }
  7439. if ($loginAsUserId != strval(intval($loginAsUserId))) {
  7440. return false;
  7441. }
  7442. // Check if the user to login is an admin
  7443. if (api_is_platform_admin_by_id($loginAsUserId)) {
  7444. // Only super admins can login to admin accounts
  7445. if (!api_global_admin_can_edit_admin($loginAsUserId)) {
  7446. return false;
  7447. }
  7448. }
  7449. $userInfo = api_get_user_info($loginAsUserId);
  7450. $isDrh = function () use ($loginAsUserId) {
  7451. if (api_is_drh()) {
  7452. if (api_drh_can_access_all_session_content()) {
  7453. $users = SessionManager::getAllUsersFromCoursesFromAllSessionFromStatus(
  7454. 'drh_all',
  7455. api_get_user_id()
  7456. );
  7457. $userList = [];
  7458. if (is_array($users)) {
  7459. foreach ($users as $user) {
  7460. $userList[] = $user['user_id'];
  7461. }
  7462. }
  7463. if (in_array($loginAsUserId, $userList)) {
  7464. return true;
  7465. }
  7466. } else {
  7467. if (api_is_drh() &&
  7468. UserManager::is_user_followed_by_drh($loginAsUserId, api_get_user_id())
  7469. ) {
  7470. return true;
  7471. }
  7472. }
  7473. }
  7474. return false;
  7475. };
  7476. $loginAsStatusForSessionAdmins = [STUDENT];
  7477. if (api_get_configuration_value('allow_session_admin_login_as_teacher')) {
  7478. $loginAsStatusForSessionAdmins[] = COURSEMANAGER;
  7479. }
  7480. return api_is_platform_admin() ||
  7481. (api_is_session_admin() && in_array($userInfo['status'], $loginAsStatusForSessionAdmins)) ||
  7482. $isDrh();
  7483. }
  7484. /**
  7485. * @return bool
  7486. */
  7487. function api_is_allowed_in_course()
  7488. {
  7489. if (api_is_platform_admin()) {
  7490. return true;
  7491. }
  7492. return Session::read('is_allowed_in_course');
  7493. }
  7494. /**
  7495. * Set the cookie to go directly to the course code $in_firstpage
  7496. * after login.
  7497. *
  7498. * @param string $value is the course code of the course to go
  7499. */
  7500. function api_set_firstpage_parameter($value)
  7501. {
  7502. setcookie('GotoCourse', $value);
  7503. }
  7504. /**
  7505. * Delete the cookie to go directly to the course code $in_firstpage
  7506. * after login.
  7507. */
  7508. function api_delete_firstpage_parameter()
  7509. {
  7510. setcookie('GotoCourse', '', time() - 3600);
  7511. }
  7512. /**
  7513. * @return bool if course_code for direct course access after login is set
  7514. */
  7515. function exist_firstpage_parameter()
  7516. {
  7517. return isset($_COOKIE['GotoCourse']) && $_COOKIE['GotoCourse'] != '';
  7518. }
  7519. /**
  7520. * @return return the course_code of the course where user login
  7521. */
  7522. function api_get_firstpage_parameter()
  7523. {
  7524. return $_COOKIE['GotoCourse'];
  7525. }
  7526. /**
  7527. * Return true on https install.
  7528. *
  7529. * @return bool
  7530. */
  7531. function api_is_https()
  7532. {
  7533. if (!empty($_SERVER['HTTP_X_FORWARDED_PROTO']) &&
  7534. $_SERVER['HTTP_X_FORWARDED_PROTO'] == 'https' || !empty($_configuration['force_https_forwarded_proto'])
  7535. ) {
  7536. $isSecured = true;
  7537. } else {
  7538. if (!empty($_SERVER['HTTPS']) && $_SERVER['HTTPS'] != 'off') {
  7539. $isSecured = true;
  7540. } else {
  7541. $isSecured = false;
  7542. // last chance
  7543. if (!empty($_SERVER['SERVER_PORT']) && $_SERVER['SERVER_PORT'] == 443) {
  7544. $isSecured = true;
  7545. }
  7546. }
  7547. }
  7548. return $isSecured;
  7549. }
  7550. /**
  7551. * Return protocol (http or https).
  7552. *
  7553. * @return string
  7554. */
  7555. function api_get_protocol()
  7556. {
  7557. return api_is_https() ? 'https' : 'http';
  7558. }
  7559. /**
  7560. * Return a string where " are replaced with 2 '
  7561. * It is useful when you pass a PHP variable in a Javascript browser dialog
  7562. * e.g. : alert("<?php get_lang('Message') ?>");
  7563. * and message contains character ".
  7564. *
  7565. * @param string $in_text
  7566. *
  7567. * @return string
  7568. */
  7569. function convert_double_quote_to_single($in_text)
  7570. {
  7571. return api_preg_replace('/"/', "''", $in_text);
  7572. }
  7573. /**
  7574. * Get origin.
  7575. *
  7576. * @param string
  7577. *
  7578. * @return string
  7579. */
  7580. function api_get_origin()
  7581. {
  7582. $origin = isset($_REQUEST['origin']) ? Security::remove_XSS($_REQUEST['origin']) : '';
  7583. return $origin;
  7584. }
  7585. /**
  7586. * Warns an user that the portal reach certain limit.
  7587. *
  7588. * @param string $limitName
  7589. */
  7590. function api_warn_hosting_contact($limitName)
  7591. {
  7592. $hostingParams = api_get_configuration_value(1);
  7593. $email = null;
  7594. if (!empty($hostingParams)) {
  7595. if (isset($hostingParams['hosting_contact_mail'])) {
  7596. $email = $hostingParams['hosting_contact_mail'];
  7597. }
  7598. }
  7599. if (!empty($email)) {
  7600. $subject = get_lang('HostingWarningReached');
  7601. $body = get_lang('PortalName').': '.api_get_path(WEB_PATH)." \n ";
  7602. $body .= get_lang('PortalLimitType').': '.$limitName." \n ";
  7603. if (isset($hostingParams[$limitName])) {
  7604. $body .= get_lang('Value').': '.$hostingParams[$limitName];
  7605. }
  7606. api_mail_html(null, $email, $subject, $body);
  7607. }
  7608. }
  7609. /**
  7610. * Gets value of a variable from app/config/configuration.php
  7611. * Variables that are not set in the configuration.php file but set elsewhere:
  7612. * - virtual_css_theme_folder (vchamilo plugin)
  7613. * - access_url (global.inc.php)
  7614. * - apc/apc_prefix (global.inc.php).
  7615. *
  7616. * @param string $variable
  7617. *
  7618. * @return bool|mixed
  7619. */
  7620. function api_get_configuration_value($variable)
  7621. {
  7622. global $_configuration;
  7623. // Check the current url id, id = 1 by default
  7624. $urlId = isset($_configuration['access_url']) ? (int) $_configuration['access_url'] : 1;
  7625. $variable = trim($variable);
  7626. // Check if variable exists
  7627. if (isset($_configuration[$variable])) {
  7628. if (is_array($_configuration[$variable])) {
  7629. // Check if it exists for the sub portal
  7630. if (array_key_exists($urlId, $_configuration[$variable])) {
  7631. return $_configuration[$variable][$urlId];
  7632. } else {
  7633. // Try to found element with id = 1 (master portal)
  7634. if (array_key_exists(1, $_configuration[$variable])) {
  7635. return $_configuration[$variable][1];
  7636. }
  7637. }
  7638. }
  7639. return $_configuration[$variable];
  7640. }
  7641. return false;
  7642. }
  7643. /**
  7644. * Returns supported image extensions in the portal.
  7645. *
  7646. * @param bool $supportVectors Whether vector images should also be accepted or not
  7647. *
  7648. * @return array Supported image extensions in the portal
  7649. */
  7650. function api_get_supported_image_extensions($supportVectors = true)
  7651. {
  7652. // jpg can also be called jpeg, jpe, jfif and jif. See https://en.wikipedia.org/wiki/JPEG#JPEG_filename_extensions
  7653. $supportedImageExtensions = ['jpg', 'jpeg', 'png', 'gif', 'jpe', 'jfif', 'jif'];
  7654. if ($supportVectors) {
  7655. array_push($supportedImageExtensions, 'svg');
  7656. }
  7657. if (version_compare(PHP_VERSION, '5.5.0', '>=')) {
  7658. array_push($supportedImageExtensions, 'webp');
  7659. }
  7660. return $supportedImageExtensions;
  7661. }
  7662. /**
  7663. * This setting changes the registration status for the campus.
  7664. *
  7665. * @author Patrick Cool <patrick.cool@UGent.be>, Ghent University
  7666. *
  7667. * @version August 2006
  7668. *
  7669. * @param bool $listCampus Whether we authorize
  7670. *
  7671. * @todo the $_settings should be reloaded here. => write api function for this and use this in global.inc.php also.
  7672. */
  7673. function api_register_campus($listCampus = true)
  7674. {
  7675. $tbl_settings = Database::get_main_table(TABLE_MAIN_SETTINGS_CURRENT);
  7676. $sql = "UPDATE $tbl_settings SET selected_value='true' WHERE variable='registered'";
  7677. Database::query($sql);
  7678. if (!$listCampus) {
  7679. $sql = "UPDATE $tbl_settings SET selected_value='true' WHERE variable='donotlistcampus'";
  7680. Database::query($sql);
  7681. }
  7682. }
  7683. /**
  7684. * Checks whether current user is a student boss.
  7685. *
  7686. * @global array $_user
  7687. *
  7688. * @return bool
  7689. */
  7690. function api_is_student_boss()
  7691. {
  7692. $_user = api_get_user_info();
  7693. return isset($_user['status']) && $_user['status'] == STUDENT_BOSS;
  7694. }
  7695. /**
  7696. * Check whether the user type should be exclude.
  7697. * Such as invited or anonymous users.
  7698. *
  7699. * @param bool $checkDB Optional. Whether check the user status
  7700. * @param int $userId Options. The user id
  7701. *
  7702. * @return bool
  7703. */
  7704. function api_is_excluded_user_type($checkDB = false, $userId = 0)
  7705. {
  7706. if ($checkDB) {
  7707. $userId = empty($userId) ? api_get_user_id() : (int) $userId;
  7708. if ($userId == 0) {
  7709. return true;
  7710. }
  7711. $userInfo = api_get_user_info($userId);
  7712. switch ($userInfo['status']) {
  7713. case INVITEE:
  7714. case ANONYMOUS:
  7715. return true;
  7716. default:
  7717. return false;
  7718. }
  7719. }
  7720. $isInvited = api_is_invitee();
  7721. $isAnonymous = api_is_anonymous();
  7722. if ($isInvited || $isAnonymous) {
  7723. return true;
  7724. }
  7725. return false;
  7726. }
  7727. /**
  7728. * Get the user status to ignore in reports.
  7729. *
  7730. * @param string $format Optional. The result type (array or string)
  7731. *
  7732. * @return array|string
  7733. */
  7734. function api_get_users_status_ignored_in_reports($format = 'array')
  7735. {
  7736. $excludedTypes = [
  7737. INVITEE,
  7738. ANONYMOUS,
  7739. ];
  7740. if ($format == 'string') {
  7741. return implode(', ', $excludedTypes);
  7742. }
  7743. return $excludedTypes;
  7744. }
  7745. /**
  7746. * Set the Site Use Cookie Warning for 1 year.
  7747. */
  7748. function api_set_site_use_cookie_warning_cookie()
  7749. {
  7750. setcookie('ChamiloUsesCookies', 'ok', time() + 31556926);
  7751. }
  7752. /**
  7753. * Return true if the Site Use Cookie Warning Cookie warning exists.
  7754. *
  7755. * @return bool
  7756. */
  7757. function api_site_use_cookie_warning_cookie_exist()
  7758. {
  7759. return isset($_COOKIE['ChamiloUsesCookies']);
  7760. }
  7761. /**
  7762. * Given a number of seconds, format the time to show hours, minutes and seconds.
  7763. *
  7764. * @param int $time The time in seconds
  7765. * @param string $originFormat Optional. PHP o JS
  7766. *
  7767. * @return string (00h00'00")
  7768. */
  7769. function api_format_time($time, $originFormat = 'php')
  7770. {
  7771. $h = get_lang('h');
  7772. $hours = $time / 3600;
  7773. $mins = ($time % 3600) / 60;
  7774. $secs = ($time % 60);
  7775. if ($time < 0) {
  7776. $hours = 0;
  7777. $mins = 0;
  7778. $secs = 0;
  7779. }
  7780. if ($originFormat == 'js') {
  7781. $formattedTime = trim(sprintf("%02d : %02d : %02d", $hours, $mins, $secs));
  7782. } else {
  7783. $formattedTime = trim(sprintf("%02d$h%02d'%02d\"", $hours, $mins, $secs));
  7784. }
  7785. return $formattedTime;
  7786. }
  7787. /**
  7788. * Create a new empty directory with index.html file.
  7789. *
  7790. * @param string $name The new directory name
  7791. * @param string $parentDirectory Directory parent directory name
  7792. *
  7793. * @return bool Return true if the directory was create. Otherwise return false
  7794. */
  7795. function api_create_protected_dir($name, $parentDirectory)
  7796. {
  7797. $isCreated = false;
  7798. if (!is_writable($parentDirectory)) {
  7799. return false;
  7800. }
  7801. $fullPath = $parentDirectory.api_replace_dangerous_char($name);
  7802. if (mkdir($fullPath, api_get_permissions_for_new_directories(), true)) {
  7803. $fp = fopen($fullPath.'/index.html', 'w');
  7804. if ($fp) {
  7805. if (fwrite($fp, '<html><head></head><body></body></html>')) {
  7806. $isCreated = true;
  7807. }
  7808. }
  7809. fclose($fp);
  7810. }
  7811. return $isCreated;
  7812. }
  7813. /**
  7814. * Sends an HTML email using the phpmailer class (and multipart/alternative to downgrade gracefully)
  7815. * Sender name and email can be specified, if not specified
  7816. * name and email of the platform admin are used.
  7817. *
  7818. * @author Bert Vanderkimpen ICT&O UGent
  7819. * @author Yannick Warnier <yannick.warnier@beeznest.com>
  7820. *
  7821. * @param string name of recipient
  7822. * @param string email of recipient
  7823. * @param string email subject
  7824. * @param string email body
  7825. * @param string sender name
  7826. * @param string sender e-mail
  7827. * @param array $extra_headers in form $headers = array($name => $value) to allow parsing
  7828. * @param array $data_file (path and filename)
  7829. * @param bool $embedded_image True for attaching a embedded file inside content html (optional)
  7830. * @param array $additionalParameters
  7831. * @param string $sendErrorTo If there's an error while sending the email, $sendErrorTo will receive a notification
  7832. *
  7833. * @return int true if mail was sent
  7834. *
  7835. * @see class.phpmailer.php
  7836. */
  7837. function api_mail_html(
  7838. $recipient_name,
  7839. $recipient_email,
  7840. $subject,
  7841. $message,
  7842. $senderName = '',
  7843. $senderEmail = '',
  7844. $extra_headers = [],
  7845. $data_file = [],
  7846. $embedded_image = false,
  7847. $additionalParameters = [],
  7848. $sendErrorTo = ''
  7849. ) {
  7850. global $platform_email;
  7851. $mail = new PHPMailer();
  7852. $mail->Mailer = $platform_email['SMTP_MAILER'];
  7853. $mail->Host = $platform_email['SMTP_HOST'];
  7854. $mail->Port = $platform_email['SMTP_PORT'];
  7855. $mail->CharSet = isset($platform_email['SMTP_CHARSET']) ? $platform_email['SMTP_CHARSET'] : 'UTF-8';
  7856. // Stay far below SMTP protocol 980 chars limit.
  7857. $mail->WordWrap = 200;
  7858. if ($platform_email['SMTP_AUTH']) {
  7859. $mail->SMTPAuth = 1;
  7860. $mail->Username = $platform_email['SMTP_USER'];
  7861. $mail->Password = $platform_email['SMTP_PASS'];
  7862. if (isset($platform_email['SMTP_SECURE'])) {
  7863. $mail->SMTPSecure = $platform_email['SMTP_SECURE'];
  7864. }
  7865. }
  7866. $mail->SMTPDebug = isset($platform_email['SMTP_DEBUG']) ? $platform_email['SMTP_DEBUG'] : 0;
  7867. // 5 = low, 1 = high
  7868. $mail->Priority = 3;
  7869. $mail->SMTPKeepAlive = true;
  7870. api_set_noreply_and_from_address_to_mailer(
  7871. $mail,
  7872. ['name' => $senderName, 'email' => $senderEmail],
  7873. !empty($extra_headers['reply_to']) ? $extra_headers['reply_to'] : []
  7874. );
  7875. if (!empty($sendErrorTo) && PHPMailer::ValidateAddress($sendErrorTo)) {
  7876. $mail->AddCustomHeader('Errors-To: '.$sendErrorTo);
  7877. }
  7878. unset($extra_headers['reply_to']);
  7879. $mail->Subject = $subject;
  7880. $mail->AltBody = strip_tags(
  7881. str_replace('<br />', "\n", api_html_entity_decode($message))
  7882. );
  7883. $list = api_get_configuration_value('send_all_emails_to');
  7884. if (!empty($list) && isset($list['emails'])) {
  7885. foreach ($list['emails'] as $email) {
  7886. $mail->AddAddress($email);
  7887. }
  7888. }
  7889. // Send embedded image.
  7890. if ($embedded_image) {
  7891. // Get all images html inside content.
  7892. preg_match_all("/<img\s+.*?src=[\"\']?([^\"\' >]*)[\"\']?[^>]*>/i", $message, $m);
  7893. // Prepare new tag images.
  7894. $new_images_html = [];
  7895. $i = 1;
  7896. if (!empty($m[1])) {
  7897. foreach ($m[1] as $image_path) {
  7898. $real_path = realpath($image_path);
  7899. $filename = basename($image_path);
  7900. $image_cid = $filename.'_'.$i;
  7901. $encoding = 'base64';
  7902. $image_type = mime_content_type($real_path);
  7903. $mail->AddEmbeddedImage(
  7904. $real_path,
  7905. $image_cid,
  7906. $filename,
  7907. $encoding,
  7908. $image_type
  7909. );
  7910. $new_images_html[] = '<img src="cid:'.$image_cid.'" />';
  7911. $i++;
  7912. }
  7913. }
  7914. // Replace origin image for new embedded image html.
  7915. $x = 0;
  7916. if (!empty($m[0])) {
  7917. foreach ($m[0] as $orig_img) {
  7918. $message = str_replace($orig_img, $new_images_html[$x], $message);
  7919. $x++;
  7920. }
  7921. }
  7922. }
  7923. $mailView = new Template(null, false, false, false, false, false, false);
  7924. $noReply = api_get_setting('noreply_email_address');
  7925. if (!empty($noReply)) {
  7926. $message .= '<br />'.get_lang('ThisIsAutomaticEmailNoReply');
  7927. }
  7928. $mailView->assign('content', $message);
  7929. if (isset($additionalParameters['link'])) {
  7930. $mailView->assign('link', $additionalParameters['link']);
  7931. }
  7932. $mailView->assign('mail_header_style', api_get_configuration_value('mail_header_style'));
  7933. $mailView->assign('mail_content_style', api_get_configuration_value('mail_content_style'));
  7934. $layout = $mailView->get_template('mail/mail.tpl');
  7935. $mail->Body = $mailView->fetch($layout);
  7936. // Attachment.
  7937. if (!empty($data_file)) {
  7938. foreach ($data_file as $file_attach) {
  7939. if (!empty($file_attach['path']) && !empty($file_attach['filename'])) {
  7940. $mail->AddAttachment($file_attach['path'], $file_attach['filename']);
  7941. }
  7942. }
  7943. }
  7944. // Only valid addresses are accepted.
  7945. if (is_array($recipient_email)) {
  7946. foreach ($recipient_email as $dest) {
  7947. if (api_valid_email($dest)) {
  7948. $mail->AddAddress($dest, $recipient_name);
  7949. }
  7950. }
  7951. } else {
  7952. if (api_valid_email($recipient_email)) {
  7953. $mail->AddAddress($recipient_email, $recipient_name);
  7954. } else {
  7955. return 0;
  7956. }
  7957. }
  7958. if (is_array($extra_headers) && count($extra_headers) > 0) {
  7959. foreach ($extra_headers as $key => $value) {
  7960. switch (strtolower($key)) {
  7961. case 'encoding':
  7962. case 'content-transfer-encoding':
  7963. $mail->Encoding = $value;
  7964. break;
  7965. case 'charset':
  7966. $mail->Charset = $value;
  7967. break;
  7968. case 'contenttype':
  7969. case 'content-type':
  7970. $mail->ContentType = $value;
  7971. break;
  7972. default:
  7973. $mail->AddCustomHeader($key.':'.$value);
  7974. break;
  7975. }
  7976. }
  7977. } else {
  7978. if (!empty($extra_headers)) {
  7979. $mail->AddCustomHeader($extra_headers);
  7980. }
  7981. }
  7982. // WordWrap the html body (phpMailer only fixes AltBody) FS#2988
  7983. $mail->Body = $mail->WrapText($mail->Body, $mail->WordWrap);
  7984. // Send the mail message.
  7985. if (!$mail->Send()) {
  7986. error_log('ERROR: mail not sent to '.$recipient_name.' ('.$recipient_email.') because of '.$mail->ErrorInfo.'<br />');
  7987. if ($mail->SMTPDebug) {
  7988. error_log(
  7989. "Connection details :: ".
  7990. "Protocol: ".$mail->Mailer.' :: '.
  7991. "Host/Port: ".$mail->Host.':'.$mail->Port.' :: '.
  7992. "Authent/Open: ".($mail->SMTPAuth ? 'Authent' : 'Open').' :: '.
  7993. ($mail->SMTPAuth ? " User/Pass: ".$mail->Username.':'.$mail->Password : '').' :: '.
  7994. "Sender: ".$mail->Sender
  7995. );
  7996. }
  7997. return 0;
  7998. }
  7999. if (!empty($additionalParameters)) {
  8000. $plugin = new AppPlugin();
  8001. $smsPlugin = $plugin->getSMSPluginLibrary();
  8002. if ($smsPlugin) {
  8003. $smsPlugin->send($additionalParameters);
  8004. }
  8005. }
  8006. // Clear all the addresses.
  8007. $mail->ClearAddresses();
  8008. // Clear all attachments
  8009. $mail->ClearAttachments();
  8010. return 1;
  8011. }
  8012. /**
  8013. * @param string $tool Possible values: GroupManager::GROUP_TOOL_*
  8014. * @param bool $showHeader
  8015. */
  8016. function api_protect_course_group($tool, $showHeader = true)
  8017. {
  8018. $groupId = api_get_group_id();
  8019. if (!empty($groupId)) {
  8020. $userId = api_get_user_id();
  8021. $groupInfo = GroupManager::get_group_properties($groupId);
  8022. // Group doesn't exists
  8023. if (empty($groupInfo)) {
  8024. api_not_allowed($showHeader);
  8025. }
  8026. // Check group access
  8027. $allow = GroupManager::user_has_access(
  8028. $userId,
  8029. $groupInfo['iid'],
  8030. $tool
  8031. );
  8032. if (!$allow) {
  8033. api_not_allowed($showHeader);
  8034. }
  8035. }
  8036. }
  8037. /**
  8038. * Check if a date is in a date range.
  8039. *
  8040. * @param datetime $startDate
  8041. * @param datetime $endDate
  8042. * @param datetime $currentDate
  8043. *
  8044. * @return bool true if date is in rage, false otherwise
  8045. */
  8046. function api_is_date_in_date_range($startDate, $endDate, $currentDate = null)
  8047. {
  8048. $startDate = strtotime(api_get_local_time($startDate));
  8049. $endDate = strtotime(api_get_local_time($endDate));
  8050. $currentDate = strtotime(api_get_local_time($currentDate));
  8051. if ($currentDate >= $startDate && $currentDate <= $endDate) {
  8052. return true;
  8053. }
  8054. return false;
  8055. }
  8056. /**
  8057. * Eliminate the duplicates of a multidimensional array by sending the key.
  8058. *
  8059. * @param array $array multidimensional array
  8060. * @param int $key key to find to compare
  8061. *
  8062. * @return array
  8063. */
  8064. function api_unique_multidim_array($array, $key)
  8065. {
  8066. $temp_array = [];
  8067. $i = 0;
  8068. $key_array = [];
  8069. foreach ($array as $val) {
  8070. if (!in_array($val[$key], $key_array)) {
  8071. $key_array[$i] = $val[$key];
  8072. $temp_array[$i] = $val;
  8073. }
  8074. $i++;
  8075. }
  8076. return $temp_array;
  8077. }
  8078. /**
  8079. * Limit the access to Session Admins when the limit_session_admin_role
  8080. * configuration variable is set to true.
  8081. */
  8082. function api_protect_limit_for_session_admin()
  8083. {
  8084. $limitAdmin = api_get_setting('limit_session_admin_role');
  8085. if (api_is_session_admin() && $limitAdmin === 'true') {
  8086. api_not_allowed(true);
  8087. }
  8088. }
  8089. /**
  8090. * Limits that a session admin has access to list users.
  8091. * When limit_session_admin_list_users configuration variable is set to true.
  8092. */
  8093. function api_protect_session_admin_list_users()
  8094. {
  8095. $limitAdmin = api_get_configuration_value('limit_session_admin_list_users');
  8096. if (api_is_session_admin() && true === $limitAdmin) {
  8097. api_not_allowed(true);
  8098. }
  8099. }
  8100. /**
  8101. * @return bool
  8102. */
  8103. function api_is_student_view_active()
  8104. {
  8105. $studentView = Session::read('studentview');
  8106. return $studentView == 'studentview';
  8107. }
  8108. /**
  8109. * Adds a file inside the upload/$type/id.
  8110. *
  8111. * @param string $type
  8112. * @param array $file
  8113. * @param int $itemId
  8114. * @param string $cropParameters
  8115. *
  8116. * @return array|bool
  8117. */
  8118. function api_upload_file($type, $file, $itemId, $cropParameters = '')
  8119. {
  8120. $upload = process_uploaded_file($file);
  8121. if ($upload) {
  8122. $name = api_replace_dangerous_char($file['name']);
  8123. // No "dangerous" files
  8124. $name = disable_dangerous_file($name);
  8125. $pathId = '/'.substr((string) $itemId, 0, 1).'/'.$itemId.'/';
  8126. $path = api_get_path(SYS_UPLOAD_PATH).$type.$pathId;
  8127. if (!is_dir($path)) {
  8128. mkdir($path, api_get_permissions_for_new_directories(), true);
  8129. }
  8130. $pathToSave = $path.$name;
  8131. $result = moveUploadedFile($file, $pathToSave);
  8132. if ($result) {
  8133. if (!empty($cropParameters)) {
  8134. $image = new Image($pathToSave);
  8135. $image->crop($cropParameters);
  8136. }
  8137. return ['path_to_save' => $pathId.$name];
  8138. }
  8139. return false;
  8140. }
  8141. }
  8142. /**
  8143. * @param string $type
  8144. * @param int $itemId
  8145. * @param string $file
  8146. *
  8147. * @return bool
  8148. */
  8149. function api_get_uploaded_web_url($type, $itemId, $file)
  8150. {
  8151. return api_get_uploaded_file($type, $itemId, $file, true);
  8152. }
  8153. /**
  8154. * @param string $type
  8155. * @param int $itemId
  8156. * @param string $file
  8157. * @param bool $getUrl
  8158. *
  8159. * @return bool
  8160. */
  8161. function api_get_uploaded_file($type, $itemId, $file, $getUrl = false)
  8162. {
  8163. $itemId = (int) $itemId;
  8164. $pathId = '/'.substr((string) $itemId, 0, 1).'/'.$itemId.'/';
  8165. $path = api_get_path(SYS_UPLOAD_PATH).$type.$pathId;
  8166. $file = basename($file);
  8167. $file = $path.'/'.$file;
  8168. if (Security::check_abs_path($file, $path) && is_file($file) && file_exists($file)) {
  8169. if ($getUrl) {
  8170. return str_replace(api_get_path(SYS_UPLOAD_PATH), api_get_path(WEB_UPLOAD_PATH), $file);
  8171. }
  8172. return $file;
  8173. }
  8174. return false;
  8175. }
  8176. /**
  8177. * @param string $type
  8178. * @param int $itemId
  8179. * @param string $file
  8180. * @param string $title
  8181. */
  8182. function api_download_uploaded_file($type, $itemId, $file, $title = '')
  8183. {
  8184. $file = api_get_uploaded_file($type, $itemId, $file);
  8185. if ($file) {
  8186. if (Security::check_abs_path($file, api_get_path(SYS_UPLOAD_PATH).$type)) {
  8187. DocumentManager::file_send_for_download($file, true, $title);
  8188. exit;
  8189. }
  8190. }
  8191. api_not_allowed(true);
  8192. }
  8193. /**
  8194. * @param string $type
  8195. * @param string $file
  8196. */
  8197. function api_remove_uploaded_file($type, $file)
  8198. {
  8199. $typePath = api_get_path(SYS_UPLOAD_PATH).$type;
  8200. $path = $typePath.'/'.$file;
  8201. if (Security::check_abs_path($path, $typePath) && file_exists($path) && is_file($path)) {
  8202. unlink($path);
  8203. }
  8204. }
  8205. /**
  8206. * @param string $type
  8207. * @param int $itemId
  8208. * @param string $file
  8209. *
  8210. * @return bool
  8211. */
  8212. function api_remove_uploaded_file_by_id($type, $itemId, $file)
  8213. {
  8214. $file = api_get_uploaded_file($type, $itemId, $file, false);
  8215. $typePath = api_get_path(SYS_UPLOAD_PATH).$type;
  8216. if (Security::check_abs_path($file, $typePath) && file_exists($file) && is_file($file)) {
  8217. unlink($file);
  8218. return true;
  8219. }
  8220. return false;
  8221. }
  8222. /**
  8223. * Converts string value to float value.
  8224. *
  8225. * 3.141516 => 3.141516
  8226. * 3,141516 => 3.141516
  8227. *
  8228. * @todo WIP
  8229. *
  8230. * @param string $number
  8231. *
  8232. * @return float
  8233. */
  8234. function api_float_val($number)
  8235. {
  8236. $number = (float) str_replace(',', '.', trim($number));
  8237. return $number;
  8238. }
  8239. /**
  8240. * Converts float values
  8241. * Example if $decimals = 2.
  8242. *
  8243. * 3.141516 => 3.14
  8244. * 3,141516 => 3,14
  8245. *
  8246. * @param string $number number in iso code
  8247. * @param int $decimals
  8248. * @param string $decimalSeparator
  8249. * @param string $thousandSeparator
  8250. *
  8251. * @return bool|string
  8252. */
  8253. function api_number_format($number, $decimals = 0, $decimalSeparator = '.', $thousandSeparator = ',')
  8254. {
  8255. $number = api_float_val($number);
  8256. return number_format($number, $decimals, $decimalSeparator, $thousandSeparator);
  8257. }
  8258. /**
  8259. * Set location url with a exit break by default.
  8260. *
  8261. * @param $url
  8262. * @param bool $exit
  8263. */
  8264. function location($url, $exit = true)
  8265. {
  8266. header('Location: '.$url);
  8267. if ($exit) {
  8268. exit;
  8269. }
  8270. }
  8271. /**
  8272. * @return string
  8273. */
  8274. function api_get_web_url()
  8275. {
  8276. if (api_get_setting('server_type') === 'test') {
  8277. return api_get_path(WEB_PATH).'web/app_dev.php/';
  8278. } else {
  8279. return api_get_path(WEB_PATH).'web/';
  8280. }
  8281. }
  8282. /**
  8283. * @param string $from
  8284. * @param string $to
  8285. *
  8286. * @return string
  8287. */
  8288. function api_get_relative_path($from, $to)
  8289. {
  8290. // some compatibility fixes for Windows paths
  8291. $from = is_dir($from) ? rtrim($from, '\/').'/' : $from;
  8292. $to = is_dir($to) ? rtrim($to, '\/').'/' : $to;
  8293. $from = str_replace('\\', '/', $from);
  8294. $to = str_replace('\\', '/', $to);
  8295. $from = explode('/', $from);
  8296. $to = explode('/', $to);
  8297. $relPath = $to;
  8298. foreach ($from as $depth => $dir) {
  8299. // find first non-matching dir
  8300. if ($dir === $to[$depth]) {
  8301. // ignore this directory
  8302. array_shift($relPath);
  8303. } else {
  8304. // get number of remaining dirs to $from
  8305. $remaining = count($from) - $depth;
  8306. if ($remaining > 1) {
  8307. // add traversals up to first matching dir
  8308. $padLength = (count($relPath) + $remaining - 1) * -1;
  8309. $relPath = array_pad($relPath, $padLength, '..');
  8310. break;
  8311. } else {
  8312. $relPath[0] = './'.$relPath[0];
  8313. }
  8314. }
  8315. }
  8316. return implode('/', $relPath);
  8317. }
  8318. /**
  8319. * Unserialize content using Brummann\Polyfill\Unserialize.
  8320. *
  8321. * @param string $type
  8322. * @param string $serialized
  8323. * @param bool $ignoreErrors. Optional.
  8324. *
  8325. * @return mixed
  8326. */
  8327. function api_unserialize_content($type, $serialized, $ignoreErrors = false)
  8328. {
  8329. switch ($type) {
  8330. case 'career':
  8331. case 'sequence_graph':
  8332. $allowedClasses = [Graph::class, VerticesMap::class, Vertices::class, Edges::class];
  8333. break;
  8334. case 'lp':
  8335. $allowedClasses = [
  8336. learnpath::class,
  8337. learnpathItem::class,
  8338. aicc::class,
  8339. aiccBlock::class,
  8340. aiccItem::class,
  8341. aiccObjective::class,
  8342. aiccResource::class,
  8343. scorm::class,
  8344. scormItem::class,
  8345. scormMetadata::class,
  8346. scormOrganization::class,
  8347. scormResource::class,
  8348. Link::class,
  8349. LpItem::class,
  8350. ];
  8351. break;
  8352. case 'course':
  8353. $allowedClasses = [
  8354. Course::class,
  8355. Announcement::class,
  8356. Attendance::class,
  8357. CalendarEvent::class,
  8358. CourseCopyLearnpath::class,
  8359. CourseCopyTestCategory::class,
  8360. CourseDescription::class,
  8361. CourseSession::class,
  8362. Document::class,
  8363. Forum::class,
  8364. ForumCategory::class,
  8365. ForumPost::class,
  8366. ForumTopic::class,
  8367. Glossary::class,
  8368. GradeBookBackup::class,
  8369. Link::class,
  8370. LinkCategory::class,
  8371. Quiz::class,
  8372. QuizQuestion::class,
  8373. QuizQuestionOption::class,
  8374. ScormDocument::class,
  8375. Survey::class,
  8376. SurveyInvitation::class,
  8377. SurveyQuestion::class,
  8378. Thematic::class,
  8379. ToolIntro::class,
  8380. Wiki::class,
  8381. Work::class,
  8382. stdClass::class,
  8383. ];
  8384. break;
  8385. case 'not_allowed_classes':
  8386. default:
  8387. $allowedClasses = false;
  8388. }
  8389. if ($ignoreErrors) {
  8390. return @Unserialize::unserialize(
  8391. $serialized,
  8392. ['allowed_classes' => $allowedClasses]
  8393. );
  8394. }
  8395. return Unserialize::unserialize(
  8396. $serialized,
  8397. ['allowed_classes' => $allowedClasses]
  8398. );
  8399. }
  8400. /**
  8401. * Set the From and ReplyTo properties to PHPMailer instance.
  8402. *
  8403. * @param PHPMailer $mailer
  8404. * @param array $sender
  8405. * @param array $replyToAddress
  8406. *
  8407. * @throws phpmailerException
  8408. */
  8409. function api_set_noreply_and_from_address_to_mailer(PHPMailer $mailer, array $sender, array $replyToAddress = [])
  8410. {
  8411. $platformEmail = $GLOBALS['platform_email'];
  8412. $noReplyAddress = api_get_setting('noreply_email_address');
  8413. $avoidReplyToAddress = false;
  8414. if (!empty($noReplyAddress)) {
  8415. $avoidReplyToAddress = api_get_configuration_value('mail_no_reply_avoid_reply_to');
  8416. }
  8417. $notification = new Notification();
  8418. // If the parameter is set don't use the admin.
  8419. $senderName = !empty($sender['name']) ? $sender['name'] : $notification->getDefaultPlatformSenderName();
  8420. $senderEmail = !empty($sender['email']) ? $sender['email'] : $notification->getDefaultPlatformSenderEmail();
  8421. // Send errors to the platform admin
  8422. $adminEmail = api_get_setting('emailAdministrator');
  8423. if (PHPMailer::ValidateAddress($adminEmail)) {
  8424. $mailer->AddCustomHeader('Errors-To: '.$adminEmail);
  8425. }
  8426. // Reply to first
  8427. if (!$avoidReplyToAddress) {
  8428. if (
  8429. !empty($replyToAddress) &&
  8430. PHPMailer::ValidateAddress($replyToAddress['mail'])
  8431. ) {
  8432. $mailer->AddReplyTo($replyToAddress['mail'], $replyToAddress['name']);
  8433. $mailer->Sender = $replyToAddress['mail'];
  8434. }
  8435. }
  8436. //If the SMTP configuration only accept one sender
  8437. if (
  8438. isset($platformEmail['SMTP_UNIQUE_SENDER']) &&
  8439. $platformEmail['SMTP_UNIQUE_SENDER']
  8440. ) {
  8441. $senderName = $platformEmail['SMTP_FROM_NAME'];
  8442. $senderEmail = $platformEmail['SMTP_FROM_EMAIL'];
  8443. if (PHPMailer::ValidateAddress($senderEmail)) {
  8444. //force-set Sender to $senderEmail, otherwise SetFrom only does it if it is currently empty
  8445. $mailer->Sender = $senderEmail;
  8446. }
  8447. }
  8448. $mailer->SetFrom($senderEmail, $senderName, !$avoidReplyToAddress);
  8449. }
  8450. /**
  8451. * @param string $template
  8452. *
  8453. * @return string
  8454. */
  8455. function api_find_template($template)
  8456. {
  8457. return Template::findTemplateFilePath($template);
  8458. }
  8459. /**
  8460. * @return array
  8461. */
  8462. function api_get_language_list_for_flag()
  8463. {
  8464. $table = Database::get_main_table(TABLE_MAIN_LANGUAGE);
  8465. $sql = "SELECT english_name, isocode FROM $table
  8466. ORDER BY original_name ASC";
  8467. static $languages = [];
  8468. if (empty($languages)) {
  8469. $result = Database::query($sql);
  8470. while ($row = Database::fetch_array($result)) {
  8471. $languages[$row['english_name']] = $row['isocode'];
  8472. }
  8473. $languages['english'] = 'gb';
  8474. }
  8475. return $languages;
  8476. }
  8477. /**
  8478. * @return string
  8479. */
  8480. function api_get_language_translate_html()
  8481. {
  8482. $translate = api_get_configuration_value('translate_html');
  8483. if (!$translate) {
  8484. return '';
  8485. }
  8486. $languageList = api_get_languages();
  8487. $hideAll = '';
  8488. foreach ($languageList['all'] as $language) {
  8489. $hideAll .= '
  8490. $("span:lang('.$language['isocode'].')").filter(
  8491. function(e, val) {
  8492. // Only find the spans if they have set the lang
  8493. if ($(this).attr("lang") == null) {
  8494. return false;
  8495. }
  8496. // Ignore ckeditor classes
  8497. return !this.className.match(/cke(.*)/);
  8498. }).hide();'."\n";
  8499. }
  8500. $userInfo = api_get_user_info();
  8501. $languageId = api_get_language_id($userInfo['language']);
  8502. $languageInfo = api_get_language_info($languageId);
  8503. $isoCode = 'en';
  8504. if (!empty($languageInfo)) {
  8505. $isoCode = $languageInfo['isocode'];
  8506. }
  8507. return '
  8508. $(function() {
  8509. '.$hideAll.'
  8510. var defaultLanguageFromUser = "'.$isoCode.'";
  8511. $("span:lang('.$isoCode.')").filter(
  8512. function() {
  8513. // Ignore ckeditor classes
  8514. return !this.className.match(/cke(.*)/);
  8515. }).show();
  8516. var defaultLanguage = "";
  8517. var langFromUserFound = false;
  8518. $(this).find("span").filter(
  8519. function() {
  8520. // Ignore ckeditor classes
  8521. return !this.className.match(/cke(.*)/);
  8522. }).each(function() {
  8523. defaultLanguage = $(this).attr("lang");
  8524. if (defaultLanguage) {
  8525. $(this).before().next("br").remove();
  8526. if (defaultLanguageFromUser == defaultLanguage) {
  8527. langFromUserFound = true;
  8528. }
  8529. }
  8530. });
  8531. // Show default language
  8532. if (langFromUserFound == false && defaultLanguage) {
  8533. $("span:lang("+defaultLanguage+")").filter(
  8534. function() {
  8535. // Ignore ckeditor classes
  8536. return !this.className.match(/cke(.*)/);
  8537. }).show();
  8538. }
  8539. });
  8540. ';
  8541. }