ÿØÿà JFIF    ÿÛ „  ( %"1!%)+...383,7(-.+  -+++--++++---+-+-----+---------------+---+-++7-----ÿÀ  ß â" ÿÄ     ÿÄ H    !1AQaq"‘¡2B±ÁÑð#R“Ò Tbr‚²á3csƒ’ÂñDS¢³$CÿÄ   ÿÄ %  !1AQa"23‘ÿÚ   ? ôÿ ¨pŸªáÿ —åYõõ\?àÒü©ŠÄï¨pŸªáÿ —åYõõ\?àÓü©ŠÄá 0Ÿªáÿ Ÿå[úƒ ú®ði~TÁbqÐ8OÕpÿ ƒOò¤Oè`–RÂáœá™êi€ßÉ< FtŸI“öÌ8úDf´°å}“¾œ6  öFá°y¥jñÇh†ˆ¢ã/ÃÐ:ªcÈ "Y¡ðÑl>ÿ ”ÏËte:qž\oäŠe÷󲍷˜HT4&ÿ ÓÐü6ö®¿øþßèô Ÿ•7Ñi’•j|“ñì>b…þS?*Óôÿ ÓÐü*h¥£ír¶ü UãS炟[AÐaè[ûª•õ&õj?†Éö+EzP—WeÒírJFt ‘BŒ†Ï‡%#tE Øz ¥OÛ«!1›üä±Í™%ºÍãö]°î(–:@<‹ŒÊö×òÆt¦ãº+‡¦%ÌÁ²h´OƒJŒtMÜ>ÀÜÊw3Y´•牋4ǍýʏTì>œú=Íwhyë,¾Ôò×õ¿ßÊa»«þˆѪQ|%6ž™A õ%:øj<>É—ÿ Å_ˆCbõ¥š±ý¯Ýƒï…¶|RëócÍf溪“t.СøTÿ *Ä¿-{†çàczůŽ_–^XþŒ±miB[X±d 1,é”zEù»& î9gœf™9Ð'.;—™i}!ôšåîqêÛ٤ёý£½ÆA–àôe"A$˝Úsäÿ ÷Û #°xŸëí(l »ý3—¥5m! rt`†0~'j2(]S¦¦kv,ÚÇ l¦øJA£Šƒ J3E8ÙiŽ:cÉžúeZ°€¯\®kÖ(79«Ž:¯X”¾³Š&¡* ….‰Ž(ÜíŸ2¥ª‡×Hi²TF¤ò[¨íÈRëÉ䢍mgÑ.Ÿ<öäS0í„ǹÁU´f#Vß;Õ–…P@3ío<ä-±»Ž.L|kªÀê›fÂ6@»eu‚|ÓaÞÆŸ…¨ááå>åŠ?cKü6ùTÍÆ”†sĤÚ;H2RÚ†õ\Ö·Ÿn'¾ ñ#ºI¤Å´%çÁ­‚â7›‹qT3Iï¨ÖÚ5I7Ë!ÅOóŸ¶øÝñØôת¦$Tcö‘[«Ö³šÒ';Aþ ¸èíg A2Z"i¸vdÄ÷.iõ®§)¿]¤À†–‡É&ä{V¶iŽ”.Ó×Õÿ û?h¬Mt–íª[ÿ Ñÿ ÌV(í}=ibÔ¡›¥¢±b Lô¥‡piη_Z<‡z§èŒ)iÖwiÇ 2hÙ3·=’d÷8éŽ1¦¸c¤µ€7›7Ø ð\á)} ¹fËí›pAÃL%âc2 í§æQz¿;T8sæ°qø)QFMð‰XŒÂ±N¢aF¨…8¯!U  Z©RÊ ÖPVÄÀÍin™Ì-GˆªÅËŠ›•zË}º±ŽÍFò¹}Uw×#ä5B¤{î}Ð<ÙD é©¤&‡ïDbàÁôMÁ." ¤‡ú*õ'VŽ|¼´Úgllº¼klz[Æüï÷Aób‡Eÿ dÑ»Xx9ÃÜ£ÁT/`¼¸vI±Ýµ·Ë‚“G³þ*Ÿû´r|*}<¨îºœ @¦mÄ’M¹”.œ«Y–|6ÏU¤jç¥ÕÞqO ˜kDÆÁ¨5ÿ š;ÐЦ¦€GÙk \ –Þ=â¼=SͧµªS°ÚÍpÜãQűÀõ¬?ÃÁ1Ñ•õZà?hóœ€ L¦l{Y*K˜Ù›zc˜–ˆâ ø+¾ ­-Ök¥%ùEÜA'}ˆ><ÊIè“bpÍ/qÞâvoX€w,\úªò6Z[XdÒæ­@Ö—€$òJí#é>'°Ú ôª˜<)4ryÙ£|óAÅn5žêŸyÒäMÝ2{"}‰–¤l÷ûWX\l¾Á¸góÉOÔ /óñB¤f¸çñ[.P˜ZsÊË*ßT܈§QN¢’¡¨§V¼(Üù*eÕ“”5T¨‹Âê¥FŒã½Dü[8'Ò¥a…Ú¶k7a *•›¼'Ò·\8¨ª\@\õ¢¦íq+DÙrmÎ…_ªæ»ŠÓœ¡¯’Ré9MÅ×D™lælffc+ŒÑ,ý™ÿ ¯þǤ=Å’Á7µ÷ÚÛ/“Ü€ñýã¼àí¾ÕÑ+ƒ,uµMâÀÄbm:ÒÎPæ{˜Gz[ƒ¯«® KHà`ߨŠéí¯P8Aq.C‰ à€kòpj´kN¶qô€…Õ,ÜNŠª-­{Zö’æû44‰sŽè‰îVíRœÕm" 6?³D9¡ÇTíÅꋇ`4«¸ÝÁô ï’ýorqКÇZ«x4Žâéþuïf¹µö[P ,Q£éaX±`PÉÍZ ¸äYúg üAx ’6Lê‚xÝÓ*äQ  Ï’¨hÍ =²,6ï#rÃ<¯–£»ƒ‹,–ê•€ aÛsñ'%Æ"®ÛüìBᝠHÚ3ß°©$“XnœÖ’î2ËTeûìxîß ¦å¿çÉ ðK§þ{‘t‚Ϋ¬jéîZ[ ”š7L¥4VÚCE×]m¤Øy”ä4-dz£œ§¸x.*ãÊÊ b÷•h:©‡¦s`BTÁRû¾g⻩‹jø sF¢àJøFl‘È•Xᓁà~*j¯ +(ÚÕ6-£¯÷GŠØy‚<Ç’.F‹Hœw(+)ÜÜâÈzÄäT§FߘãÏ;DmVœ3Àu@mÚüXÝü•3B¨òÌÁÛ<·ÃÜ z,Ì@õÅ·d2]ü8s÷IôÞ¯^Ç9¢u„~ëAŸï4«M? K]­ÅàPl@s_ p:°¬ZR”´›JC[CS.h‹ƒïËœ«Æ]–÷ó‚wR×k7X‰k›‘´ù¦=¡«‰¨¨Â')—71ó’c‡Ðúµ `é.{§p¹ój\Ž{1h{o±Ý=áUÊïGÖŒõ–-BÄm+AZX¶¡ ïHðæ¥JmÙ;…䡟ˆ¦ ° äšiÉg«$üMk5¤L“’çÊvïâï ,=f“"íἊ5ô¬x6{ɏžID0e¸vçmi'︧ºð9$ò¹÷*£’9ÿ ²TÔ…×>JV¥}Œ}$p[bÔ®*[jzS*8 ”·T›Í–ñUîƒwo$áè=LT™ç—~ô·¤ÈÚ$榍q‰„+´kFm)ž‹©i–ËqÞŠ‰à¶ü( ‚•§ •°ò·‡#5ª•µÊ﯅¡X¨šÁ*F#TXJÊ ušJVÍ&=iÄs1‚3•'fý§5Ñ<=[íÞ­ PÚ;ѱÌ_~Ä££8rÞ ²w;’hDT°>ÈG¬8Á²ÚzŽ®ò®qZcqJêäÞ-ö[ܘbň±çb“ж31²n×iƒðÕ;1¶þÉ ªX‰,ßqÏ$>•î íZ¥Z 1{ç൵+ƒÕµ¥°T$§K]á»Ûï*·¤tMI’ÂZbŽÕiÒ˜}bÓ0£ª5›¨ [5Ž^ÝœWøÂÝh° ¢OWun£¤5 a2Z.G2³YL]jåtì”ä ÁÓ‘%"©<Ôúʰsº UZvä‡ÄiÆÒM .÷V·™ø#kèýiíÌ–ª)µT[)BˆõÑ xB¾B€ÖT¨.¥~ð@VĶr#¸ü*åZNDŽH;âi ],©£öØpù(šºãö¼T.uCê•4@ÿ GÕÛ)Cx›®0ø#:ÏðFÒbR\(€€Ä®fã4Þ‰Fä¯HXƒÅ,†öEÑÔÜ]Öv²?tLÃvBY£ú6Êu5ÅAQ³1‘’¬x–HŒÐ‡ ^ ¸KwJôÖŽ5×CÚ¨vÜ«/B0$×k°=ðbÇ(Ï)w±A†Á† 11Í=èQšµ626ŒÜ/`G«µ<}—-Ö7KEHÈÉðóȤmݱû±·ø«Snmá=“䫚mݱŸ¡¶~ó·“äUóJæúòB|E LêŽy´jDÔ$G¢þÐñ7óR8ýÒ…Ç› WVe#·Ÿ p·Fx~•ݤF÷0Èÿ K¯æS<6’¡WШ; ´ÿ ¥Êø\Òuî†åÝ–VNœkÒ7oòX¨Á­Ø÷FÎÑä±g÷ÿ M~Çî=p,X´ ÝÌÚÅ‹’ÃjÖ.ØöÏñ qïQ¤ÓZE†° =6·]܈ s¸>v•Ž^Ý\wq9r‰Î\¸¡kURÒ$­*‹Nq?Þª*!sŠÆ:TU_u±T+øX¡ ®¹¡,ÄâÃBTsÜ$Ø›4m椴zÜK]’’›Pƒ @€#â˜`é¹=I‡fiV•Ôî“nRm+µFPOhÍ0B£ €+¬5c v•:P'ÒyÎ ‰V~‚Ó†ÖuókDoh$å\*ö%Ю=£«…aȼ½÷Û.-½VŒŠ¼'lyî±1¬3ó#ÞE¿ÔS¤gV£m›=§\û"—WU¤ÚǼÿ ÂnÁGŒÃ ‚õN D³õNÚíŒÕ;HôyÄÈ©P¹Ä{:?R‘Ô¨âF÷ø£bÅó® JS|‚R÷ivýáâ€Æé¡è³´IئÑT!§˜•ت‚¬â@q€wnïCWÄ@JU€ê¯m6]Ï:£âx'+ÒðXvÓ¦Úm=–´7œ $ì“B£~p%ÕŸUþ« N@¼üï~w˜ñø5®—'Ôe»¤5ã//€ž~‰Tþ›Å7•#¤× Íö pÄ$ùeåì*«ÓŠEØWEÈsßg ¦ûvžSsLpºÊW–âµEWöˬH; ™!CYõZ ÃÄf æ#1W. \uWâ\,\Çf j’<qTbên›Î[vxx£ë 'ö¨1›˜ÀM¼Pÿ H)ƒêêŒA7s,|F“ 꺸k³9Ìö*ç®;Ö!Ö$Eiž•¹ÒÚ†ýóéÝû¾ÕS®ó$’NÝäŸz¤5r¦ãÄÃD÷Üø!°ø‡Ô&@m™Ì^Ãä­d q5Lnÿ N;.6½·N|#ä"1Nƒx“ã<3('&ñßt  ~ªu”1Tb㫨9ê–›–bìd$ߣ=#ÕãÒmU¯eí$EFù5ýYô櫨æì™Ç—±ssM]·á¿0ÕåJRÓªîiƒ+O58ÖñªŠÒx" \µâá¨i’¤i —Ö ” M+M¤ë9‚‰A¦°Qõ¾ßøK~¼Ã‘g…Ö´~÷Ï[3GUœÒ½#…kàÔ®Ò”‰³·dWV‰IP‰Ú8u¹”E ÖqLj¾êÕCBš{A^Âß;–¨`¯¬ìö ˼ ×tìø.tƐm*n¨y4o&Àx¥n¦×î‡aupáÛj8¿m›è¶ã!o½;ß0y^ý×^EÑ¿ÒjzŒ­)vÚÑnÄL …^ªô× ‡—‚3k Îý­hï]içå–îÏ*÷ñþ»Ô CÒjøjÍznˆ´ ¹#b'Fô‹ ‰v¥'’à'T´ƒHýÍ%M‰ ƒ&ÆÇŒï1 ‘ –Þ ‰i¬s žR-Ÿ kЬá¬7:þ 0ŒÅÒÕ/aÙ¬ÃÝ#Úøœ ©aiVc‰. ¹¦ãµ” ›Yg¦›ÆÎýº°f³7ƒhá·¸­}&D9¡ÂsÉÙÞèŠõØàC™¨ñbFC|´Ü(ŸƒÚÒ-%»'a Ì¿)ËÇn¿úÿ ÞŽX…4ÊÅH^ôΑí@ù¹Eh¶“L8Çjù ¼ÎåVªóR©Ï5uà V4lZß®=€xÖŸ–ÑÈ ÷”¨°¾__yM1tÉ?uÆþIkÄgæ@þ[¢†°XÃJ£j·:nkÅ¢u ‘}âGzö­/IµèЬ¼48q¦F°ŽR¼=ûì{´¯RýicS ÕÛ íNtÍÙï£,w4rêì®»~x(©Uñ§#Ñ&œÕ¤>ÎåÍÓ9’Ö{9eV­[Öjâ²ãu]˜å2›qÑšÕJç0€sÄ|Êëè0튔bÁ>“{×_F`Ø©ºê:µä,v¤ðfc1±"«ÔÍän1#=· Âøv~H½ÐßA¾¿Ü€Óš]Õ; I¾÷ç‚Qi†î¹9ywÔKG˜áñ zQY—§ÃÕZ07§X‚ Áh;ÁM)iÌCH-¯T‘ë|A0{Ò½LÚ–TâÖkÜ’dÀ“rmm»”جPF³ÖcbE§T€ÒxKºû’Ó®7±²(\4ŽÃ¸Uu@j™yĵ;³µ!Á¢b.W¤=mõ´êµK k ¸K^ÜÛ#p*Ü14qkZç5ïë †°5Ï%ÍÛ<Õ¤×Ô¥ê†C Õ´¼ú$ƒÖ“”]Ù¬qÞÚ[4©ý!ûÏ—Áb쳐XµA¬â~`›Çr¸8ìùÝ䫦<>ä÷«?xs´ÇÑ /á;¹øüÊÈÙà{"@Žïzâ¬[âß‚ U_<ÇŸ½4èN˜ú61®qŠu ¦þF£»äJ_ˆÙÎ~ ÞAã–݄ϗrŠD;xTž‘ô`É«…suãO`?³à™ô Lý#Íc5öoæØ‚y´´÷«ZR§<&JÇ+éâô´€i!Àˆ0æAoàðLèÖ-2ŸõW.’t^–(KÁmHµV@xÜÇy®Ñø­â^:Ú3w· 7½¹°ñ¸â¹®:',«Mœ—n­Á+Ãbš LÈ‘ÄnRÓÅœ%¦²‰¨ùQ:¤f‚ "PÕtô¸…cæl…&˜Ú˜Ôkv‹ž+vŠ,=¢v­6—Xy*¥t£«<™:“aîϲ=¦6rO]XI¿Œ÷¤zÚ­›¶ 6÷”w\d ü~v®ˆÌk«^m<ÿ ¢‰Õ\)ùºŽ;… lîÙÅEŠ®cѾ@vnMÏ,¼“ñ•ŽBxðÃzãÇç%3ˆ"}Ù•Åî> BÉú;Ò]V+P˜F_´ßé> Øše|ï‡ÄOmFæÇ ãqÞ$/xÐx­z`ï9"œÜij‚!7.\Td…9M‡•iŽ‹¾‘50ÞŽn¥ß4ÉôO ¹*í^QêËÜÇÌ8=ާs‰'ÂëÙ«á%Pú[O †ÅP¯Vsް.‰,kc¶ ¬A9n˜XÎ-ÞšN["¹QÕ‰ƒMýÁߺXJæÍaLj¾×Ãmã¾ãÚ uñÒþåQô¦¥ /ÄUx:‚ÍÜ’ Đ©ØÝ3V¨‰ÕnÐ6ó*óúK­«…c ¯U òhsý­jóÔj#,ímŒRµ«lbïUTŒÑ8†Ä0œÏr`ð¡¬É Ї ë"À² ™ 6¥ f¶ ¢ÚoܱԷ-<Àî)†a¶ž'Ú»¨TXqØæ¶÷YÄHy˜9ÈIW­YÀuMFë ºÏ’AqÌ4·/Ú †ô'i$øä­=Ä Ý|öK×40è|È6p‘0§)o¥ctî§H+CA-“ xØ|ÐXАç l8íºð3Ø:³¤¬KX¯UÿÙ 'use strict'; /* * Engines which do not support caching of their file contents * should use the `read()` function defined in consolidate.js * On top of this, when an engine compiles to a `Function`, * these functions should either be cached within consolidate.js * or the engine itself via `options.cache`. This will allow * users and frameworks to pass `options.cache = true` for * `NODE_ENV=production`, however edit the file(s) without * re-loading the application in development. */ /** * Module dependencies. */ var fs = require('fs'); var path = require('path'); var Promise = require('bluebird'); var join = path.join; var resolve = path.resolve; var extname = path.extname; var dirname = path.dirname; var isAbsolute = path.isAbsolute; var readCache = {}; /** * Require cache. */ var cacheStore = {}; /** * Require cache. */ var requires = {}; /** * Clear the cache. * * @api public */ exports.clearCache = function() { readCache = {}; cacheStore = {}; }; /** * Conditionally cache `compiled` template based * on the `options` filename and `.cache` boolean. * * @param {Object} options * @param {Function} compiled * @return {Function} * @api private */ function cache(options, compiled) { // cachable if (compiled && options.filename && options.cache) { delete readCache[options.filename]; cacheStore[options.filename] = compiled; return compiled; } // check cache if (options.filename && options.cache) { return cacheStore[options.filename]; } return compiled; } /** * Read `path` with `options` with * callback `(err, str)`. When `options.cache` * is true the template string will be cached. * * @param {String} options * @param {Function} cb * @api private */ function read(path, options, cb) { var str = readCache[path]; var cached = options.cache && str && typeof str === 'string'; // cached (only if cached is a string and not a compiled template function) if (cached) return cb(null, str); // read fs.readFile(path, 'utf8', function(err, str) { if (err) return cb(err); // remove extraneous utf8 BOM marker str = str.replace(/^\uFEFF/, ''); if (options.cache) readCache[path] = str; cb(null, str); }); } /** * Read `path` with `options` with * callback `(err, str)`. When `options.cache` * is true the partial string will be cached. * * @param {String} options * @param {Function} fn * @api private */ function readPartials(path, options, cb) { if (!options.partials) return cb(); var partials = options.partials; var keys = Object.keys(partials); function next(index) { if (index === keys.length) return cb(null); var key = keys[index]; var partialPath = partials[key]; if (partialPath === undefined || partialPath === null || partialPath === false) { return next(++index); } var file; if (isAbsolute(partialPath)) { if (extname(partialPath) !== '') { file = partialPath; } else { file = join(partialPath + extname(path)); } } else { file = join(dirname(path), partialPath + extname(path)); } read(file, options, function(err, str) { if (err) return cb(err); options.partials[key] = str; next(++index); }); } next(0); } /** * promisify */ function promisify(cb, fn) { return new Promise(function(resolve, reject) { cb = cb || function(err, html) { if (err) { return reject(err); } resolve(html); }; fn(cb); }); } /** * fromStringRenderer */ function fromStringRenderer(name) { return function(path, options, cb) { options.filename = path; return promisify(cb, function(cb) { readPartials(path, options, function(err) { if (err) return cb(err); if (cache(options)) { exports[name].render('', options, cb); } else { read(path, options, function(err, str) { if (err) return cb(err); exports[name].render(str, options, cb); }); } }); }); }; } /** * velocity support. */ exports.velocityjs = fromStringRenderer('velocityjs'); /** * velocity string support. */ exports.velocityjs.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.velocityjs || (requires.velocityjs = require('velocityjs')); try { options.locals = options; cb(null, engine.render(str, options).trimLeft()); } catch (err) { cb(err); } }); }; /** * Liquid support. */ exports.liquid = fromStringRenderer('liquid'); /** * Liquid string support. */ /** * Note that in order to get filters and custom tags we've had to push * all user-defined locals down into @locals. However, just to make things * backwards-compatible, any property of `options` that is left after * processing and removing `locals`, `meta`, `filters`, `customTags` and * `includeDir` will also become a local. */ function _renderTinyliquid(engine, str, options, cb) { var context = engine.newContext(); var k; /** * Note that there's a bug in the library that doesn't allow us to pass * the locals to newContext(), hence looping through the keys: */ if (options.locals) { for (k in options.locals) { context.setLocals(k, options.locals[k]); } delete options.locals; } if (options.meta) { context.setLocals('page', options.meta); delete options.meta; } /** * Add any defined filters: */ if (options.filters) { for (k in options.filters) { context.setFilter(k, options.filters[k]); } delete options.filters; } /** * Set up a callback for the include directory: */ var includeDir = options.includeDir || process.cwd(); context.onInclude(function(name, callback) { var extname = path.extname(name) ? '' : '.liquid'; var filename = path.resolve(includeDir, name + extname); fs.readFile(filename, {encoding: 'utf8'}, function(err, data) { if (err) return callback(err); callback(null, engine.parse(data)); }); }); delete options.includeDir; /** * The custom tag functions need to have their results pushed back * through the parser, so set up a shim before calling the provided * callback: */ var compileOptions = { customTags: {} }; if (options.customTags) { var tagFunctions = options.customTags; for (k in options.customTags) { /*Tell jshint there's no problem with having this function in the loop */ /*jshint -W083 */ compileOptions.customTags[k] = function(context, name, body) { var tpl = tagFunctions[name](body.trim()); context.astStack.push(engine.parse(tpl)); }; /*jshint +W083 */ } delete options.customTags; } /** * Now anything left in `options` becomes a local: */ for (k in options) { context.setLocals(k, options[k]); } /** * Finally, execute the template: */ var tmpl = cache(context) || cache(context, engine.compile(str, compileOptions)); tmpl(context, cb); } exports.liquid.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.liquid; var Liquid; try { // set up tinyliquid engine engine = requires.liquid = require('tinyliquid'); // use tinyliquid engine _renderTinyliquid(engine, str, options, cb); return; } catch (err) { // set up liquid-node engine try { Liquid = requires.liquid = require('liquid-node'); engine = new Liquid.Engine(); } catch (err) { throw err; } } // use liquid-node engine try { var locals = options.locals || {}; if (options.meta) { locals.pages = options.meta; delete options.meta; } /** * Add any defined filters: */ if (options.filters) { engine.registerFilters(options.filters); delete options.filters; } /** * Set up a callback for the include directory: */ var includeDir = options.includeDir || process.cwd(); engine.fileSystem = new Liquid.LocalFileSystem(includeDir, 'liquid'); delete options.includeDir; /** * The custom tag functions need to have their results pushed back * through the parser, so set up a shim before calling the provided * callback: */ if (options.customTags) { var tagFunctions = options.customTags; for (k in options.customTags) { engine.registerTag(k, tagFunctions[k]); } delete options.customTags; } /** * Now anything left in `options` becomes a local: */ for (var k in options) { locals[k] = options[k]; } /** * Finally, execute the template: */ return engine .parseAndRender(str, locals) .nodeify(function(err, result) { if (err) { throw new Error(err); } else { return cb(null, result); } }); } catch (err) { cb(err); } }); }; /** * Jade support. */ exports.jade = function(path, options, cb) { return promisify(cb, function(cb) { var engine = requires.jade; if (!engine) { try { engine = requires.jade = require('jade'); } catch (err) { try { engine = requires.jade = require('then-jade'); } catch (otherError) { throw err; } } } try { var tmpl = cache(options) || cache(options, engine.compileFile(path, options)); cb(null, tmpl(options)); } catch (err) { cb(err); } }); }; /** * Jade string support. */ exports.jade.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.jade; if (!engine) { try { engine = requires.jade = require('jade'); } catch (err) { try { engine = requires.jade = require('then-jade'); } catch (otherError) { throw err; } } } try { var tmpl = cache(options) || cache(options, engine.compile(str, options)); cb(null, tmpl(options)); } catch (err) { cb(err); } }); }; /** * Dust support. */ exports.dust = fromStringRenderer('dust'); /** * Dust string support. */ exports.dust.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.dust; if (!engine) { try { engine = requires.dust = require('dust'); } catch (err) { try { engine = requires.dust = require('dustjs-helpers'); } catch (err) { engine = requires.dust = require('dustjs-linkedin'); } } } var ext = 'dust'; var views = '.'; if (options) { if (options.ext) ext = options.ext; if (options.views) views = options.views; if (options.settings && options.settings.views) views = options.settings.views; } if (!options || (options && !options.cache)) engine.cache = {}; engine.onLoad = function(path, callback) { if (extname(path) === '') path += '.' + ext; if (path[0] !== '/') path = views + '/' + path; read(path, options, callback); }; try { var templateName; if (options.filename) { templateName = options.filename.replace(new RegExp('^' + views + '/'), '').replace(new RegExp('\\.' + ext), ''); } var tmpl = cache(options) || cache(options, engine.compileFn(str, templateName)); tmpl(options, cb); } catch (err) { cb(err); } }); }; /** * Swig support. */ exports.swig = fromStringRenderer('swig'); /** * Swig string support. */ exports.swig.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.swig; if (!engine) { try { engine = requires.swig = require('swig'); } catch (err) { try { engine = requires.swig = require('swig-templates'); } catch (otherError) { throw err; } } } try { if (options.cache === true) options.cache = 'memory'; engine.setDefaults({ cache: options.cache }); var tmpl = cache(options) || cache(options, engine.compile(str, options)); cb(null, tmpl(options)); } catch (err) { cb(err); } }); }; /** * Atpl support. */ exports.atpl = fromStringRenderer('atpl'); /** * Atpl string support. */ exports.atpl.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.atpl || (requires.atpl = require('atpl')); try { var tmpl = cache(options) || cache(options, engine.compile(str, options)); cb(null, tmpl(options)); } catch (err) { cb(err); } }); }; /** * Liquor support, */ exports.liquor = fromStringRenderer('liquor'); /** * Liquor string support. */ exports.liquor.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.liquor || (requires.liquor = require('liquor')); try { var tmpl = cache(options) || cache(options, engine.compile(str, options)); cb(null, tmpl(options)); } catch (err) { cb(err); } }); }; /** * Twig support. */ exports.twig = fromStringRenderer('twig'); /** * Twig string support. */ exports.twig.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.twig || (requires.twig = require('twig').twig); var templateData = { data: str }; try { var tmpl = cache(templateData) || cache(templateData, engine(templateData)); cb(null, tmpl.render(options)); } catch (err) { cb(err); } }); }; /** * EJS support. */ exports.ejs = fromStringRenderer('ejs'); /** * EJS string support. */ exports.ejs.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.ejs || (requires.ejs = require('ejs')); try { var tmpl = cache(options) || cache(options, engine.compile(str, options)); cb(null, tmpl(options)); } catch (err) { cb(err); } }); }; /** * Eco support. */ exports.eco = fromStringRenderer('eco'); /** * Eco string support. */ exports.eco.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.eco || (requires.eco = require('eco')); try { cb(null, engine.render(str, options)); } catch (err) { cb(err); } }); }; /** * Jazz support. */ exports.jazz = fromStringRenderer('jazz'); /** * Jazz string support. */ exports.jazz.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.jazz || (requires.jazz = require('jazz')); try { var tmpl = cache(options) || cache(options, engine.compile(str, options)); tmpl.eval(options, function(str) { cb(null, str); }); } catch (err) { cb(err); } }); }; /** * JQTPL support. */ exports.jqtpl = fromStringRenderer('jqtpl'); /** * JQTPL string support. */ exports.jqtpl.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.jqtpl || (requires.jqtpl = require('jqtpl')); try { engine.template(str, str); cb(null, engine.tmpl(str, options)); } catch (err) { cb(err); } }); }; /** * Haml support. */ exports.haml = fromStringRenderer('haml'); /** * Haml string support. */ exports.haml.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.haml || (requires.haml = require('hamljs')); try { options.locals = options; cb(null, engine.render(str, options).trimLeft()); } catch (err) { cb(err); } }); }; /** * Hamlet support. */ exports.hamlet = fromStringRenderer('hamlet'); /** * Hamlet string support. */ exports.hamlet.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.hamlet || (requires.hamlet = require('hamlet')); try { options.locals = options; cb(null, engine.render(str, options).trimLeft()); } catch (err) { cb(err); } }); }; /** * Whiskers support. */ exports.whiskers = function(path, options, cb) { return promisify(cb, function(cb) { var engine = requires.whiskers || (requires.whiskers = require('whiskers')); engine.__express(path, options, cb); }); }; /** * Whiskers string support. */ exports.whiskers.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.whiskers || (requires.whiskers = require('whiskers')); try { cb(null, engine.render(str, options)); } catch (err) { cb(err); } }); }; /** * Coffee-HAML support. */ exports['haml-coffee'] = fromStringRenderer('haml-coffee'); /** * Coffee-HAML string support. */ exports['haml-coffee'].render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires['haml-coffee'] || (requires['haml-coffee'] = require('haml-coffee')); try { var tmpl = cache(options) || cache(options, engine.compile(str, options)); cb(null, tmpl(options)); } catch (err) { cb(err); } }); }; /** * Hogan support. */ exports.hogan = fromStringRenderer('hogan'); /** * Hogan string support. */ exports.hogan.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.hogan || (requires.hogan = require('hogan.js')); try { var tmpl = cache(options) || cache(options, engine.compile(str, options)); cb(null, tmpl.render(options, options.partials)); } catch (err) { cb(err); } }); }; /** * templayed.js support. */ exports.templayed = fromStringRenderer('templayed'); /** * templayed.js string support. */ exports.templayed.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.templayed || (requires.templayed = require('templayed')); try { var tmpl = cache(options) || cache(options, engine(str)); cb(null, tmpl(options)); } catch (err) { cb(err); } }); }; /** * Handlebars support. */ exports.handlebars = fromStringRenderer('handlebars'); /** * Handlebars string support. */ exports.handlebars.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.handlebars || (requires.handlebars = require('handlebars')); try { for (var partial in options.partials) { engine.registerPartial(partial, options.partials[partial]); } for (var helper in options.helpers) { engine.registerHelper(helper, options.helpers[helper]); } var tmpl = cache(options) || cache(options, engine.compile(str, options)); cb(null, tmpl(options)); } catch (err) { cb(err); } }); }; /** * Underscore support. */ exports.underscore = fromStringRenderer('underscore'); /** * Underscore string support. */ exports.underscore.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.underscore || (requires.underscore = require('underscore')); try { for (var partial in options.partials) { options.partials[partial] = engine.template(options.partials[partial]); } var tmpl = cache(options) || cache(options, engine.template(str, null, options)); cb(null, tmpl(options).replace(/\n$/, '')); } catch (err) { cb(err); } }); }; /** * Lodash support. */ exports.lodash = fromStringRenderer('lodash'); /** * Lodash string support. */ exports.lodash.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.lodash || (requires.lodash = require('lodash')); try { var tmpl = cache(options) || cache(options, engine.template(str, options)); cb(null, tmpl(options).replace(/\n$/, '')); } catch (err) { cb(err); } }); }; /** * Pug support. (formerly Jade) */ exports.pug = function(path, options, cb) { return promisify(cb, function(cb) { var engine = requires.pug; if (!engine) { try { engine = requires.pug = require('pug'); } catch (err) { try { engine = requires.pug = require('then-pug'); } catch (otherError) { throw err; } } } try { var tmpl = cache(options) || cache(options, engine.compileFile(path, options)); cb(null, tmpl(options)); } catch (err) { cb(err); } }); }; /** * Pug string support. */ exports.pug.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.pug; if (!engine) { try { engine = requires.pug = require('pug'); } catch (err) { try { engine = requires.pug = require('then-pug'); } catch (otherError) { throw err; } } } try { var tmpl = cache(options) || cache(options, engine.compile(str, options)); cb(null, tmpl(options)); } catch (err) { cb(err); } }); }; /** * QEJS support. */ exports.qejs = fromStringRenderer('qejs'); /** * QEJS string support. */ exports.qejs.render = function(str, options, cb) { return promisify(cb, function(cb) { try { var engine = requires.qejs || (requires.qejs = require('qejs')); engine.render(str, options).then(function(result) { cb(null, result); }, function(err) { cb(err); }).done(); } catch (err) { cb(err); } }); }; /** * Walrus support. */ exports.walrus = fromStringRenderer('walrus'); /** * Walrus string support. */ exports.walrus.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.walrus || (requires.walrus = require('walrus')); try { var tmpl = cache(options) || cache(options, engine.parse(str)); cb(null, tmpl.compile(options)); } catch (err) { cb(err); } }); }; /** * Mustache support. */ exports.mustache = fromStringRenderer('mustache'); /** * Mustache string support. */ exports.mustache.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.mustache || (requires.mustache = require('mustache')); try { cb(null, engine.to_html(str, options, options.partials)); } catch (err) { cb(err); } }); }; /** * Just support. */ exports.just = function(path, options, cb) { return promisify(cb, function(cb) { var engine = requires.just; if (!engine) { var JUST = require('just'); engine = requires.just = new JUST(); } engine.configure({ useCache: options.cache }); engine.render(path, options, cb); }); }; /** * Just string support. */ exports.just.render = function(str, options, cb) { return promisify(cb, function(cb) { var JUST = require('just'); var engine = new JUST({ root: { page: str }}); engine.render('page', options, cb); }); }; /** * ECT support. */ exports.ect = function(path, options, cb) { return promisify(cb, function(cb) { var engine = requires.ect; if (!engine) { var ECT = require('ect'); engine = requires.ect = new ECT(options); } engine.configure({ cache: options.cache }); engine.render(path, options, cb); }); }; /** * ECT string support. */ exports.ect.render = function(str, options, cb) { return promisify(cb, function(cb) { var ECT = require('ect'); var engine = new ECT({ root: { page: str }}); engine.render('page', options, cb); }); }; /** * mote support. */ exports.mote = fromStringRenderer('mote'); /** * mote string support. */ exports.mote.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.mote || (requires.mote = require('mote')); try { var tmpl = cache(options) || cache(options, engine.compile(str)); cb(null, tmpl(options)); } catch (err) { cb(err); } }); }; /** * Toffee support. */ exports.toffee = function(path, options, cb) { return promisify(cb, function(cb) { var toffee = requires.toffee || (requires.toffee = require('toffee')); toffee.__consolidate_engine_render(path, options, cb); }); }; /** * Toffee string support. */ exports.toffee.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.toffee || (requires.toffee = require('toffee')); try { engine.str_render(str, options, cb); } catch (err) { cb(err); } }); }; /** * doT support. */ exports.dot = fromStringRenderer('dot'); /** * doT string support. */ exports.dot.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.dot || (requires.dot = require('dot')); var extend = (requires.extend || (requires.extend = require('util')._extend)); try { var settings = {}; settings = extend(settings, engine.templateSettings); settings = extend(settings, options ? options.dot : {}); var tmpl = cache(options) || cache(options, engine.template(str, settings, options)); cb(null, tmpl(options)); } catch (err) { cb(err); } }); }; /** * bracket support. */ exports.bracket = fromStringRenderer('bracket'); /** * bracket string support. */ exports.bracket.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.bracket || (requires.bracket = require('bracket-template')); try { var tmpl = cache(options) || cache(options, engine.default.compile(str, options)); cb(null, tmpl(options)); } catch (err) { cb(err); } }); }; /** * Ractive support. */ exports.ractive = fromStringRenderer('ractive'); /** * Ractive string support. */ exports.ractive.render = function(str, options, cb) { return promisify(cb, function(cb) { var Engine = requires.ractive || (requires.ractive = require('ractive')); var template = cache(options) || cache(options, Engine.parse(str)); options.template = template; if (options.data === null || options.data === undefined) { var extend = (requires.extend || (requires.extend = require('util')._extend)); // Shallow clone the options object options.data = extend({}, options); // Remove consolidate-specific properties from the clone var i; var length; var properties = ['template', 'filename', 'cache', 'partials']; for (i = 0, length = properties.length; i < length; i++) { var property = properties[i]; delete options.data[property]; } } try { cb(null, new Engine(options).toHTML()); } catch (err) { cb(err); } }); }; /** * Nunjucks support. */ exports.nunjucks = fromStringRenderer('nunjucks'); /** * Nunjucks string support. */ exports.nunjucks.render = function(str, options, cb) { return promisify(cb, function(cb) { try { var engine = options.nunjucksEnv || requires.nunjucks || (requires.nunjucks = require('nunjucks')); var env = engine; // deprecated fallback support for express // // if (options.settings && options.settings.views) { env = engine.configure(options.settings.views); } else if (options.nunjucks && options.nunjucks.configure) { env = engine.configure.apply(engine, options.nunjucks.configure); } // // because `renderString` does not initiate loaders // we must manually create a loader for it based off // either `options.settings.views` or `options.nunjucks` or `options.nunjucks.root` // // // // // so instead we simply check if we passed a custom loader // otherwise we create a simple file based loader if (options.loader) { env = new engine.Environment(options.loader); } else if (options.settings && options.settings.views) { env = new engine.Environment( new engine.FileSystemLoader(options.settings.views) ); } else if (options.nunjucks && options.nunjucks.loader) { if (typeof options.nunjucks.loader === 'string') { env = new engine.Environment(new engine.FileSystemLoader(options.nunjucks.loader)); } else { env = new engine.Environment( new engine.FileSystemLoader( options.nunjucks.loader[0], options.nunjucks.loader[1] ) ); } } env.renderString(str, options, cb); } catch (err) { throw cb(err); } }); }; /** * HTMLing support. */ exports.htmling = fromStringRenderer('htmling'); /** * HTMLing string support. */ exports.htmling.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.htmling || (requires.htmling = require('htmling')); try { var tmpl = cache(options) || cache(options, engine.string(str)); cb(null, tmpl.render(options)); } catch (err) { cb(err); } }); }; /** * Rendering function */ function requireReact(module, filename) { var babel = requires.babel || (requires.babel = require('babel-core')); var compiled = babel.transformFileSync(filename, { presets: [ 'react' ] }).code; return module._compile(compiled, filename); } exports.requireReact = requireReact; /** * Converting a string into a node module. */ function requireReactString(src, filename) { var babel = requires.babel || (requires.babel = require('babel-core')); if (!filename) filename = ''; var m = new module.constructor(); filename = filename || ''; // Compile Using React var compiled = babel.transform(src, { presets: [ 'react' ] }).code; // Compile as a module m.paths = module.paths; m._compile(compiled, filename); return m.exports; } /** * A naive helper to replace {{tags}} with options.tags content */ function reactBaseTmpl(data, options) { var exp; var regex; // Iterates through the keys in file object // and interpolate / replace {{key}} with it's value for (var k in options) { if (options.hasOwnProperty(k)) { exp = '{{' + k + '}}'; regex = new RegExp(exp, 'g'); if (data.match(regex)) { data = data.replace(regex, options[k]); } } } return data; } /** * Plates Support. */ exports.plates = fromStringRenderer('plates'); /** * Plates string support. */ exports.plates.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.plates || (requires.plates = require('plates')); var map = options.map || undefined; try { var tmpl = engine.bind(str, options, map); cb(null, tmpl); } catch (err) { cb(err); } }); }; /** * The main render parser for React bsaed templates */ function reactRenderer(type) { if (require.extensions) { // Ensure JSX is transformed on require if (!require.extensions['.jsx']) { require.extensions['.jsx'] = requireReact; } // Supporting .react extension as well as test cases // Using .react extension is not recommended. if (!require.extensions['.react']) { require.extensions['.react'] = requireReact; } } // Return rendering fx return function(str, options, cb) { return promisify(cb, function(cb) { // React Import var ReactDOM = requires.ReactDOM || (requires.ReactDOM = require('react-dom/server')); var react = requires.react || (requires.react = require('react')); // Assign HTML Base var base = options.base; delete options.base; var enableCache = options.cache; delete options.cache; var isNonStatic = options.isNonStatic; delete options.isNonStatic; // Start Conversion try { var Code; var Factory; var baseStr; var content; var parsed; if (!cache(options)) { // Parsing if (type === 'path') { var path = resolve(str); delete require.cache[path]; Code = require(path); } else { Code = requireReactString(str); } Factory = cache(options, react.createFactory(Code)); } else { Factory = cache(options); } parsed = new Factory(options); content = (isNonStatic) ? ReactDOM.renderToString(parsed) : ReactDOM.renderToStaticMarkup(parsed); if (base) { baseStr = readCache[str] || fs.readFileSync(resolve(base), 'utf8'); if (enableCache) { readCache[str] = baseStr; } options.content = content; content = reactBaseTmpl(baseStr, options); } cb(null, content); } catch (err) { cb(err); } }); }; } /** * React JS Support */ exports.react = reactRenderer('path'); /** * React JS string support. */ exports.react.render = reactRenderer('string'); /** * ARC-templates support. */ exports['arc-templates'] = fromStringRenderer('arc-templates'); /** * ARC-templates string support. */ exports['arc-templates'].render = function(str, options, cb) { var readFileWithOptions = Promise.promisify(read); var consolidateFileSystem = {}; consolidateFileSystem.readFile = function(path) { return readFileWithOptions(path, options); }; return promisify(cb, function(cb) { try { var engine = requires['arc-templates']; if (!engine) { var Engine = require('arc-templates/dist/es5'); engine = requires['arc-templates'] = new Engine({ filesystem: consolidateFileSystem }); } var compiler = cache(options) || cache(options, engine.compileString(str, options.filename)); compiler.then(function(func) { return func(options); }) .then(function(result) { cb(null, result.content); }) .catch(cb); } catch (err) { cb(err); } }); }; /** * Vash support */ exports.vash = fromStringRenderer('vash'); /** * Vash string support */ exports.vash.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.vash || (requires.vash = require('vash')); try { // helper system : https://github.com/kirbysayshi/vash#helper-system if (options.helpers) { for (var key in options.helpers) { if (!options.helpers.hasOwnProperty(key) || typeof options.helpers[key] !== 'function') { continue; } engine.helpers[key] = options.helpers[key]; } } var tmpl = cache(options) || cache(options, engine.compile(str, options)); tmpl(options, function sealLayout(err, ctx) { if (err) cb(err); ctx.finishLayout(); cb(null, ctx.toString().replace(/\n$/, '')); }); } catch (err) { cb(err); } }); }; /** * Slm support. */ exports.slm = fromStringRenderer('slm'); /** * Slm string support. */ exports.slm.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.slm || (requires.slm = require('slm')); try { var tmpl = cache(options) || cache(options, engine.compile(str, options)); cb(null, tmpl(options)); } catch (err) { cb(err); } }); }; /** * Marko support. */ exports.marko = function(path, options, cb) { return promisify(cb, function(cb) { var engine = requires.marko || (requires.marko = require('marko')); options.writeToDisk = !!options.cache; try { var tmpl = cache(options) || cache(options, engine.load(path, options)); tmpl.renderToString(options, cb); } catch (err) { cb(err); } }); }; /** * Marko string support. */ exports.marko.render = function(str, options, cb) { return promisify(cb, function(cb) { var engine = requires.marko || (requires.marko = require('marko')); options.writeToDisk = !!options.cache; options.filename = options.filename || 'string.marko'; try { var tmpl = cache(options) || cache(options, engine.load(options.filename, str, options)); tmpl.renderToString(options, cb); } catch (err) { cb(err); } }); }; /** * Teacup support. */ exports.teacup = function(path, options, cb) { return promisify(cb, function(cb) { var engine = requires.teacup || (requires.teacup = require('teacup/lib/express')); require.extensions['.teacup'] = require.extensions['.coffee']; if (path[0] !== '/') { path = join(process.cwd(), path); } if (!options.cache) { var callback = cb; cb = function() { delete require.cache[path]; callback.apply(this, arguments); }; } engine.renderFile(path, options, cb); }); }; /** * Teacup string support. */ exports.teacup.render = function(str, options, cb) { var coffee = require('coffee-script'); var vm = require('vm'); var sandbox = { module: {exports: {}}, require: require }; return promisify(cb, function(cb) { vm.runInNewContext(coffee.compile(str), sandbox); var tmpl = sandbox.module.exports; cb(null, tmpl(options)); }); }; /** * expose the instance of the engine */ exports.requires = requires;