From ef41eac5ac9348b4f1fb223d9c4ab9e22effb985 Mon Sep 17 00:00:00 2001 From: yulong Date: Mon, 29 Sep 2025 14:54:48 +0000 Subject: [PATCH] =?UTF-8?q?=E4=B8=8A=E4=BC=A0=E6=96=87=E4=BB=B6=E8=87=B3?= =?UTF-8?q?=20jtxtv10/lib?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- jtxtv10/lib/hm.json | 22 ++ jtxtv10/lib/huya2.js | 1 + jtxtv10/lib/ip.json | 1 + jtxtv10/lib/jinja.js | 577 +++++++++++++++++++++++++++++++++++++++++++ 4 files changed, 601 insertions(+) create mode 100644 jtxtv10/lib/hm.json create mode 100644 jtxtv10/lib/huya2.js create mode 100644 jtxtv10/lib/ip.json create mode 100644 jtxtv10/lib/jinja.js diff --git a/jtxtv10/lib/hm.json b/jtxtv10/lib/hm.json new file mode 100644 index 0000000..27a3d03 --- /dev/null +++ b/jtxtv10/lib/hm.json @@ -0,0 +1,22 @@ +{ + "SiteUrl": "https://www.haimianxz.com,https://www.hmxz.org,https://www.5536806.xyz,http://haimian.eu.org", + "Classes": [ + { + "type_name": "电影", + "type_id": "1" + }, + { + "type_name": "剧集", + "type_id": "2" + }, + { + "type_name": "综艺", + "type_id": "5" + }, + { + "type_name": "动漫", + "type_id": "3" + } + ], + "Cookie": "bbs_sid=nll6gb39qj1e4fgkvsjh930ti5; _xn_accesscount_visited=1; __51uvsct__K2MEuBiiS0iv1Gsb=1; __51vcke__K2MEuBiiS0iv1Gsb=df02f20c-b83d-507b-b310-c3f8ab36a87a; __51vuft__K2MEuBiiS0iv1Gsb=1745249876903; bbs_token=rhHYDxTHFdsKEOsD84q8NBkSi6aTY8KlfnqpjudmsOxKBcJe; __vtins__K2MEuBiiS0iv1Gsb=%7B%22sid%22%3A%20%2299dcbd93-0d23-5a24-abb5-ec507c6c248a%22%2C%20%22vd%22%3A%2017%2C%20%22stt%22%3A%20464771%2C%20%22dr%22%3A%201425%2C%20%22expires%22%3A%201745251199999%2C%20%22ct%22%3A%201745250341665%7D" +} \ No newline at end of file diff --git a/jtxtv10/lib/huya2.js b/jtxtv10/lib/huya2.js new file mode 100644 index 0000000..11aceac --- /dev/null +++ b/jtxtv10/lib/huya2.js @@ -0,0 +1 @@ +**dmFyIHJ1bGU9e3RpdGxlOiLomY7niZnnm7Tmkq0iLGhvc3Q6Imh0dHBzOi8vd3d3Lmh1eWEuY29tIixob21lVXJsOiIvY2FjaGUucGhwP209TGl2ZUxpc3QmZG89Z2V0TGl2ZUxpc3RCeVBhZ2UmZ2FtZUlkPTIxNjgmdGFnQWxsPTAmcGFnZT0xIix1cmw6Ii9jYWNoZS5waHA/bT1MaXZlTGlzdCZkbz1nZXRMaXZlTGlzdEJ5UGFnZSZnYW1lSWQ9ZnlmaWx0ZXImdGFnQWxsPTAmcGFnZT1meXBhZ2UiLGNsYXNzX25hbWU6IuWoseS5kCbnvZHmuLgm5Y2V5py6JuaJi+a4uCIsY2xhc3NfdXJsOiI4JjEmMiYzIixkZXRhaWxVcmw6Imh0dHBzOi8vbS5odXlhLmNvbS9meWlkIixmaWx0ZXJhYmxlOjEsZmlsdGVyX3VybDoie3tmbC5jYXRlSWR9fSIsZmlsdGVyX2RlZjp7ODp7Y2F0ZUlkOiIyMTM1In0sMTp7Y2F0ZUlkOiIxIn0sMjp7Y2F0ZUlkOiIxNzMyIn0sMzp7Y2F0ZUlkOiIyMzM2In19LGZpbHRlcjp7ODpbe2tleToiY2F0ZUlkIixuYW1lOiLliIbnsbsiLHZhbHVlOlt7bjoi5pif56eAIix2OiIxNjYzIn0se246IuaIt+WkliIsdjoiMjE2NSJ9LHtuOiLkuIDotbfnnIsiLHY6IjIxMzUifSx7bjoi5LqM5qyh5YWDIix2OiIyNjMzIn0se246IuiZmuaLn+WBtuWDjyIsdjoiNjA1NSJ9LHtuOiLml4XmuLgiLHY6IjY3OTEifSx7bjoi5pS+5pig5Y6FIix2OiI2MjQ1In0se246IuWoseS5kOWkqeWcsCIsdjoiMTAwMDIyIn0se246IuS6pOWPiyIsdjoiNDA3OSJ9LHtuOiLnu4TpmJ8iLHY6IjUzNjcifSx7bjoi5ZCD5Zad546p5LmQIix2OiIxMDAwNDQifSx7bjoi5Y6f5YibIix2OiI2ODYxIn0se246IuiZjueJmeaWh+WMliIsdjoiNDA4OSJ9LHtuOiLkvZPogrIiLHY6IjIzNTYifSx7bjoi6JmO54mZ5Zyw5pa5Iix2OiI1MTIzIn0se246IuminOWAvCIsdjoiMjE2OCJ9LHtuOiLnp5HmioAiLHY6IjI0MDgifSx7bjoi6Z+z5LmQIix2OiIzNzkzIn0se246Iui2o+WIhuS6qyIsdjoiNTg4MyJ9LHtuOiLkuIDotbfkubAiLHY6Ijc3NTkifSx7bjoi5rS+5a+5Iix2OiI3Nzg1In1dfV0sMTpbe2tleToiY2F0ZUlkIixuYW1lOiLliIbnsbsiLHZhbHVlOlt7bjoi6Iux6ZuE6IGU55ufIix2OiIxIn0se246IkNTMiIsdjoiODYyIn0se246Iuepv+i2iueBq+e6vyIsdjoiNCJ9LHtuOiJsb2zkupHpobbkuYvlvIgiLHY6IjU0ODUifSx7bjoi5peg55WP5aWR57qmIix2OiI1OTM3In0se246IkNGSEQiLHY6IjYwNzkifSx7bjoi6YCG5oiYIix2OiIxMzUifSx7bjoi54KJ55+z5Lyg6K+0Iix2OiIzOTMifSx7bjoiRE9UQTIiLHY6IjcifSx7bjoi5Zyw5LiL5Z+O5LiO5YuH5aOrIix2OiIyIn0se246IumtlOWFveS4lueVjCIsdjoiOCJ9LHtuOiLlnablhYvkuJbnlYwiLHY6IjgwMiJ9LHtuOiJET1RBMSIsdjoiNiJ9LHtuOiLmoqbkuInlm70iLHY6IjQ4OSJ9LHtuOiLprZTlhb3kuonpnLgzIix2OiI0NjE1In0se246IumXrumBkyIsdjoiMTA3In0se246IuWRvei/kOaWueiInyIsdjoiMzA1OCJ9LHtuOiJRUemjnui9piIsdjoiOSJ9LHtuOiLmmJ/pmYXkuonpnLgiLHY6IjUifSx7bjoi572R5ri456ue5oqAIix2OiIxMDAwMjMifSx7bjoi5bCE5Ye757u85ZCI5ri45oiPIix2OiIxMDAxNDEifSx7bjoi5pq06Zuq5LiT5Yy6Iix2OiIxMDAwNDMifSx7bjoi5b2p6Jm55bKbT25saW5lIix2OiI2ODMifSx7bjoi5YmR54G1Iix2OiI4OTcifSx7bjoi5Yab5LqL5ri45oiPIix2OiIxMDAxMzMifSx7bjoi5YaS6Zmp5bKbIix2OiIyMjQzIn0se246Iuaal+m7keegtOWdj+elniIsdjoiMTEyMyJ9LHtuOiLor5vku5kzIix2OiIxNjQ2In0se246IueDreihgOaxn+a5liIsdjoiMzg3In0se246IuiLsemtguS5i+WIgyIsdjoiMTgzMCJ9LHtuOiLmrabmnpflpJbkvKDkuIDkuJbnkLTnvJgiLHY6IjE2NjEifSx7bjoi6LW35Yeh77ya576k6ZuE6YCQ6bm/Iix2OiIxNjEyIn0se246IuelnuatpjTnlLXohJHniYgiLHY6IjMyMjcifSx7bjoi6b6Z5LmL6LC3Iix2OiIxNSJ9LHtuOiLngonnn7PmiJjmo4siLHY6IjU3NTEifSx7bjoi5b6h6b6Z5Zyo5aSpIix2OiI2NzUifSx7bjoi6YCG5rC05a+SIix2OiIyOTUyIn0se246IuWNg+W5tDMiLHY6Ijg3OCJ9LHtuOiLmsLjmgZLkuYvloZQiLHY6IjQ0NiJ9LHtuOiLkvZPogrLmuLjmiI8iLHY6IjEwMDEzNSJ9LHtuOiLlr7vku5kiLHY6IjczNCJ9LHtuOiLmiJjoiLDkuJbnlYwiLHY6IjE5NDcifSx7bjoiUVHkuInlm70iLHY6IjEwOTAifSx7bjoi5rWB5pS+5LmL6LevIix2OiI0MjcifSx7bjoi5Y+N5oGQ57K+6IuxT25saW5lIix2OiIxOTE4In0se246IuWPjeaBkOihjOWKqG9ubGluZSIsdjoiODYxIn0se246IuW+gemAlCIsdjoiMjcxNSJ9LHtuOiLmiJjkuonpm7fpnIYiLHY6IjYyNCJ9LHtuOiLkuJ3ot6/kvKDor7QyIix2OiIxMDI2In0se246IuaYn+mZheaImOeUsiIsdjoiNjI3In0se246Ik5CQTJLT0zns7vliJciLHY6IjM5NTkifSx7bjoi5Lmd6Zi055yf57uPIix2OiIxMDA5In0se246Iui3kei3keWNoeS4gei9piIsdjoiMTYyIn0se246Iuivm+S7meS4lueVjCIsdjoiNzc0OSJ9LHtuOiJRUeWNjuWkjyIsdjoiMTg3OCJ9LHtuOiLlpKnmtq/mmI7mnIjliIAiLHY6IjEyMTkifSx7bjoi5aSn6K+d6KW/5ri477ya5b2S5p2lIix2OiI4MjM5In0se246IuiNkumHjuihjOWKqFBD54mIIix2OiIzMTg1In0se246IuaWsOWJkeS+oOaDhee8mCIsdjoiNTg2In0se246IuWJkee9kTMiLHY6IjkwMCJ9LHtuOiLnlJ/mrbvni5nlh7syIix2OiI2MDkxIn0se246IuWFqOeQg+S9v+WRvSIsdjoiOTM5In0se246IuaipuaDs+S4lueVjDMiLHY6IjQ4NiJ9LHtuOiLmnqrnpZ7nuqoiLHY6IjQ5NiJ9LHtuOiLmlrDlpKnpvpnlhavpg6giLHY6IjU2NzEifSx7bjoi6ZOB55Sy6ZuE5YW1Iix2OiIyNzY1In0se246IuelnuazoyIsdjoiMjUzMSJ9LHtuOiLmlpfmiJjnpZ4iLHY6IjU5MSJ9LHtuOiLpgKDmoqbopb/muLhPTCIsdjoiNjgxNSJ9LHtuOiLlpKnloIIiLHY6IjE5NjYifSx7bjoi5aSn6K+d6KW/5ri4MiIsdjoiMjk3NSJ9LHtuOiLkvb/lkb3lj6zllKTvvJrmiJjljLoiLHY6IjU5MTEifSx7bjoi5biM5pybT0wiLHY6IjExNjEifSx7bjoi5p6B5YWJ5LiW55WMIOW8keelnuS8oCIsdjoiNTE0In0se246IuWuiOacm+WFiOmUi+W9kuadpSIsdjoiMjE3NCJ9LHtuOiJRUeiHqueUseW5u+aDsyIsdjoiMTg2MiJ9LHtuOiLlkb3ov5AyIix2OiIyOTQyIn0se246IuWlh+i/ueS4lueVjDIiLHY6Ijc2NiJ9LHtuOiLlnablhYvlpKfmiJgiLHY6IjQzNTkifSx7bjoi55Sf5q2754uZ5Ye7Iix2OiIyNDcxIn0se246IuWkqee/vOWGsyIsdjoiNzc5In0se246IlFR5bm75oOzIix2OiIyNDE5In0se246IuaWsOmjnumjnihGbHlGRikiLHY6IjE1ODIifSx7bjoi5YiA5YmR6Iux6ZuEIix2OiI5MTUifSx7bjoiRklGQSBPbmxpbmXns7vliJciLHY6IjEwMDA3OSJ9LHtuOiLlhajnkIPkvb/lkb0zIix2OiIyOTUzIn0se246IuWujOe+juerr+a4uOezu+WIlyIsdjoiMzAzNCJ9LHtuOiLmiJjmhI8iLHY6IjI1OTkifSx7bjoi5rOh5rOh5aCCIix2OiI0NDAifSx7bjoi6LWb5bCU5Y+3Iix2OiIyNzU1In0se246IuWkp+WUkOaXoOWPjOmbtiIsdjoiMTU4NCJ9LHtuOiJRUeeCq+iIniIsdjoiMjQ0MCJ9LHtuOiIwMDfvvJrkvKDlpYciLHY6IjExMzUifSx7bjoi5aSp5LiLIix2OiIxNTk3In0se246IuWkqeiwlSIsdjoiMTg5OSJ9LHtuOiLmlrDlgKnlpbPlub3prYIiLHY6IjE1NzkifSx7bjoi5Lyg5LiW5peg5Y+MIix2OiI5ODQifSx7bjoi5YmR5L6g5LiW55WMIix2OiI5MDMifSx7bjoiV2FyaGF2ZW4iLHY6IjkwNTMifSx7bjoi6K+65Lqa5Lyg6K+0Iix2OiIxOTAifSx7bjoi5paw5oyR5oiYIix2OiI1ODMifSx7bjoi6LaF5r+A5paX5qKm5aKDIix2OiI1NjkxIn0se246IlFR6Z+z6YCfIix2OiIxMDg1In0se246IuW+gemAlDIiLHY6IjY3NyJ9LHtuOiLlvoHnqIsiLHY6IjY3OCJ9LHtuOiLonIDpl6giLHY6IjQ3MTEifSx7bjoi5a6M576O5LiW55WM77ya6K+456We5LmL5oiYIix2OiI3MjE3In0se246IuaImOS5i+WIg++8muW5uOWtmOiAhSIsdjoiODYyNyJ9LHtuOiLpu47mmI7kuYvlhYkiLHY6IjQxIn0se246IuWkqeWRveilv+a4uCIsdjoiMTA4NiJ9LHtuOiLngqvoiJ7ml7bku6MiLHY6IjMzNTMifSx7bjoi6I2j6KqJ56m66Ze0Iix2OiIyMjUifSx7bjoi5Y2h5ouJ5b285LiYIix2OiI5MDczIn0se246IuaIkOWQieaAneaxl+aAgOaXp+eJiCIsdjoiNjQwIn0se246IuiLsembhOW5tOS7ozMiLHY6IjEyMzIifSx7bjoi5aSp5Lmm5aWH6LCIIix2OiIyMjI1In0se246IuWKsuiInuWboiIsdjoiMjQyMCJ9LHtuOiLov5zlvoFPbmxpbmXmoqbmg7PniYgiLHY6IjE0MiJ9LHtuOiLlpYfov7nvvJrkvKDlpYciLHY6IjM5MTcifSx7bjoi5bCB5Y2w6ICFIix2OiIyMjk0In0se246IuS4iuWPpOS4lue6qiIsdjoiMTA0NiJ9LHtuOiLmoqblubvor5vku5kyIix2OiI0ODgifSx7bjoiVEVSQSBPbmxpbmUiLHY6IjEwNzIifSx7bjoi5YCa5aSp4oWhIix2OiI5NTkifSx7bjoi6KGX5aS056+u55CDIix2OiIyMDYifSx7bjoi5rC45oGS6L2u5Zue77ya5peg6ZmQIix2OiI3ODQ3In0se246IueBq+e6v+eyvuiLsSIsdjoiMjU1MCJ9LHtuOiLlv43ogIXmnZHlpKfmiJgyIix2OiIyMzY5In0se246IumihuWcsOS6uueUnyIsdjoiMjI4MiJ9LHtuOiLku5nkvqDkuJbnlYwiLHY6Ijk5NSJ9LHtuOiLmtJvlpYfoi7Hpm4TkvKAiLHY6IjQzMiJ9LHtuOiJLQVJEUyIsdjoiODI2MSJ9LHtuOiLml6DpmZDms5XliJkiLHY6IjMxODkifSx7bjoi5YWo6Z2i5oiY5LqJ77ya56ue5oqA5Zy6Iix2OiI1OTAxIn0se246IuiHqueUseevrueQgyIsdjoiMjQzIn0se246IkZDIE9ubGluZTQiLHY6IjM2ODMifSx7bjoi5oiY5Zyw5LmL546LIix2OiI2MTgifSx7bjoi5Y+k5YmR5aWH6LCtT0wiLHY6IjE4OTIifSx7bjoiUVHku5nkvqDkvKAiLHY6IjIyOTEifSx7bjoiMzAw6Iux6ZuEIix2OiIxMTMyIn1dfV0sMjpbe2tleToiY2F0ZUlkIixuYW1lOiLliIbnsbsiLHZhbHVlOlt7bjoi5aSp5aSp5ZCD6bihIix2OiIyNzkzIn0se246IuawuOWKq+aXoOmXtCIsdjoiNjIxOSJ9LHtuOiLkuLvmnLrmuLjmiI8iLHY6IjEwMDAzMiJ9LHtuOiLnjJvlhb3mtL7lr7kiLHY6IjYxNjUifSx7bjoi5LqS5Yqo54K55pKtIix2OiI1OTA3In0se246IuaIkeeahOS4lueVjCIsdjoiMTczMiJ9LHtuOiLmlrnoiJ8iLHY6IjE5OTcifSx7bjoi5Y2V5py654Ot5ri4Iix2OiIxMDAwMDIifSx7bjoi5oCA5pen5ri45oiPIix2OiIxMDAxMjUifSx7bjoi6YCD56a75aGU56eR5aSrIix2OiIzNDkzIn0se246IuS/hOe9l+aWr+mSk+mxvDQiLHY6IjU0OTUifSx7bjoi6YOo6JC977ya5LiK5Y2HIix2OiIxMzE4In0se246IkRhcmsgYW5kIERhcmtlciIsdjoiNzkwNSJ9LHtuOiJUaGUgRnJvbnQiLHY6Ijk0OTcifSx7bjoiQXBleOiLsembhCIsdjoiNTAxMSJ9LHtuOiLnlJ/ljJbljbHmnLo06YeN5Yi254mIIix2OiI4MDEzIn0se246IkRheVrni6znq4vniYgiLHY6IjExMjUifSx7bjoi5pif56m6Iix2OiI3ODU3In0se246IuS6kuWKqOWJp+a4uCIsdjoiNjkxOSJ9LHtuOiLoib7lsJTnmbvms5Xnjq8iLHY6IjU4MDEifSx7bjoi5oGQ5oOn5LmL6Ze0Iix2OiI2Njc5In0se246IlNDVU0iLHY6IjQyNDUifSx7bjoiRHJlYWQgSHVuZ2VyIix2OiI3NjAxIn0se246IuWhnuWwlOi+vuS8oOivtO+8mueOi+WbveS5i+azqiIsdjoiNzg4MyJ9LHtuOiLmrKfmtLLljaHovabmqKHmi58iLHY6IjQ3NSJ9LHtuOiLmtJvlhYvnjovlm70iLHY6IjI4NjQifSx7bjoi5Y2n6b6Z77ya6IuN5aSp6Zmo6JC9Iix2OiI3ODU5In0se246IuaXoOS6uua3seepuiIsdjoiMjU2NiJ9LHtuOiLluJ3lm73npZ7or50iLHY6IjY4MjEifSx7bjoi6aWl6I2SIix2OiI3NCJ9LHtuOiLmo67mnpfkuYvlrZAiLHY6Ijc5NDMifSx7bjoi5pif55CD5aSn5oiY57O75YiXIix2OiI1NTQifSx7bjoi5p6B6ZmQ56ue6YCf77ya5Zyw5bmz57q/Iix2OiIyNjM0In0se246IuacgOe7iOW5u+aDszE2Iix2OiI3ODY5In0se246IkFtb25nIFVzIix2OiI2MTYzIn0se246IuaAqueJqeeMjuS6uu+8muW0m+i1tyIsdjoiNjQ3OSJ9LHtuOiLmgKrniannjI7kurrnianor60iLHY6IjcxMDEifSx7bjoi6aqR6ams5LiO56CN5p2A57O75YiXIix2OiI0NzgzIn0se246IuiNkumHjuWkp+mVluWuojIiLHY6IjQzMTkifSx7bjoi5aSq6I2S5Yid5aKDIix2OiI3Njg1In0se246Iue9l+W4g+S5kOaAnSIsdjoiNTc3MSJ9LHtuOiLlj6rni7zvvJrlvbHpgJ3kuozluqYiLHY6IjQ1MDUifSx7bjoi5Y+M5Lq65oiQ6KGMIix2OiI2NzM3In0se246IumHjeeUn+i+uee8mCIsdjoiNjIwMSJ9LHtuOiLmtbfotLznjosg5a+756eY5LiW55WMIix2OiI1MDk3In0se246IuaImOelnu+8muivuOelnum7hOaYjyIsdjoiNzc3MSJ9LHtuOiLmgZDprLznl4ciLHY6IjYyMDUifSx7bjoi6ay86LC35YWr6I2SIix2OiI2NTcxIn0se246IumcjeagvOayg+iMqOS5i+mBlyIsdjoiNzg4MSJ9LHtuOiLlhajpnaLmiJjkuokiLHY6IjM1MjEifSx7bjoi5LuB546LMiIsdjoiNTc5NSJ9LHtuOiLlpaXmi4nmmJ8iLHY6IjI4NDYifSx7bjoi55Sc6Jyc5LmL5a62Iix2OiI2NzM5In0se246IuS7meWJkeWlh+S+oOS8oOS4gyIsdjoiNjUwOSJ9LHtuOiLmtojpgJ3nmoTlhYnoipIyIix2OiI3NTgxIn0se246Iua4oeelnuiusCIsdjoiNjIzMSJ9LHtuOiLlvZLlrrblvILpgJQiLHY6IjI5NDkifSx7bjoi5Y2a5b635LmL6ZeoMyIsdjoiNjE0NyJ9LHtuOiLnnIvpl6jni5fvvJrlhpvlm6IiLHY6IjYxNTUifSx7bjoi5L2/5ZG95Y+s5ZSk57O75YiXIix2OiIxMDAxMzcifSx7bjoi6IKy56Kn5ri45oiPIix2OiIxMDAxMzkifSx7bjoi5bid5Zu95pe25LujNCIsdjoiNDgzNSJ9LHtuOiLoi7HngbXnpZ7mrr8iLHY6IjY2MDkifSx7bjoi6Juu6I2S5b+X5byC77ya6LW35rqQIix2OiI5NzA5In0se246IuWymyIsdjoiMzAxOSJ9LHtuOiLmspnnm5LkuI7lia/mnKwiLHY6IjkxNTEifSx7bjoi5omr6Zu3Iix2OiIyMzQ5In0se246Iuihl+acuua4uOaIjyIsdjoiNTk5OSJ9LHtuOiLlub3ngbXnur/vvJrkuJzkuqwiLHY6Ijc2NjkifSx7bjoi5oCq54mp54yO5Lq65LiW55WMIix2OiIzNTE5In0se246Iuebl+i0vOS5i+a1tyIsdjoiMzY0MSJ9LHtuOiLnurjkuroiLHY6IjUyNTcifSx7bjoi6buR6Imy5YOP57Sg5Lq6Iix2OiIyOCJ9LHtuOiLlr7npqazlspvkuYvprYIiLHY6IjYwMzkifSx7bjoi55Ge5aWH5LiO5Y+u5b2TIix2OiIyNDU1In0se246Iua4r+ivoeWunuW9lSIsdjoiNTg1MyJ9LHtuOiLliZHlo6siLHY6IjE0NjcifSx7bjoi6a2U5rOV6Zeo5LmL6Iux6ZuE5peg5pWM57O75YiXIix2OiIyMDk2In0se246IuaYn+mcsuiwt+eJqeivrSIsdjoiMjQ0MyJ9LHtuOiLku5nliZHlpYfkvqDkvKDlm5siLHY6IjE2NTkifSx7bjoi5Lyk5a6z5LiW55WMIix2OiIyMzM3In0se246IuemgemXreaxgueUnyIsdjoiNjA2NSJ9LHtuOiLnnJ/kuInlm73ml6Dlj4wiLHY6IjE1OTkifSx7bjoi5oGQ5oCW6buO5piOIix2OiIzNDM1In0se246IuWFtuS7luWNleacuiIsdjoiMzA2OSJ9LHtuOiLlub3ngbXooYzliqjvvJrojZLph44iLHY6IjI3OTQifSx7bjoi6L6557yYIix2OiIxNTEifSx7bjoi6Zi/5Yeh6L6+Iix2OiIxMDYifSx7bjoi5YWo6Z2i5oiY5LqJ77ya5LiJ5Zu9Iix2OiIzMDA0In0se246Ium7juaYjuS5i+aYnyIsdjoiNDAifSx7bjoi57+85pif5rGC55SfIix2OiI3NDYzIn0se246Ium7juaYjuS8oOivtCIsdjoiMzkifSx7bjoi5oG26a2U5LmL6a2CIix2OiI2MTUxIn0se246IuiJvuWFsOWymyIsdjoiMzg1NSJ9LHtuOiLpvpnkuI7lnLDkuIvln47vvJrnp5jlvbHljobku6PorrBIRCIsdjoiMTcifSx7bjoi6b6Z5LiO5Zyw5LiL5Z+O77ya5YyV6aaW6LC3Iix2OiIxOCJ9LHtuOiLmspnnn7PplYfml7blhYkiLHY6Ijc2ODcifSx7bjoi5LiJ5Zu95b+X5pu55pON5LygIix2OiIyNTkyIn0se246IkZJRkHotrPnkIMiLHY6IjE4ODgifSx7bjoi5pyA57uI5bm75oOzN++8mumHjeWItueJiCIsdjoiNTgwOSJ9LHtuOiLlm5vmtbflhYTlvJ8iLHY6IjU5OTUifSx7bjoi5pyA57uI5bm75oOz77ya6LW35rqQIix2OiI3NjUzIn0se246IuaRlOi3pOWfjuWkp+S5seaWlyIsdjoiNzc3MyJ9LHtuOiLpn7PkuZDmuLjmiI8iLHY6IjI3NjEifSx7bjoi57K+54G15LiO6JCk54Gr5oSP5b+XIix2OiI1ODk1In0se246IuahpeaigeW7uumAoOW4iCIsdjoiMzI3NSJ9LHtuOiLlk4jov6rmlq8iLHY6IjYxNTMifSx7bjoi5bCP57yH5aic55qE5aWH5bm75LmL5ZywIix2OiI3NjQ3In0se246IuWuneWPr+aipu+8muWJkeebviIsdjoiNTcxNSJ9LHtuOiLovrnlooMiLHY6IjQ3NzkifSx7bjoi5Lq657G75LiA6LSl5raC5ZywIix2OiIzMjg5In0se246IuezluixhuS6uu+8mue7iOaegea3mOaxsOi1myIsdjoiNjA4MyJ9LHtuOiLnsr7ngbXlrp3lj6/moqZMZXQncyBHbyIsdjoiNDM3NSJ9LHtuOiLmiJjplKQ0MEvvvJrmmpfmva4iLHY6IjMwMTYifSx7bjoi55+z5rK56aqa5YqoIix2OiIyNTg1In0se246Iue0q+Whnueni+mjjiIsdjoiNjA0NSJ9LHtuOiLpmL/lsJTmr5TmgakiLHY6IjgxMTUifSx7bjoi54yO5Lq677ya6I2S6YeO55qE5Y+s5ZSkIix2OiIyOTA2In0se246IkZJRkEgMjMiLHY6Ijc3ODkifSx7bjoi6YO95biC77ya5aSp6ZmF57q/Iix2OiIyMjAxIn0se246IuS5kOmrmOWkp+eUteW9se+8mua4uOaIj+eJiCIsdjoiMTQzOSJ9LHtuOiLlhajpnaLmiJjkuonvvJrpmL/mj5Dmi4kiLHY6IjIzODMifSx7bjoi5pap5aaW6KGMIix2OiI2MTA1In0se246IumprOmHjOWlpei1m+i9pjgiLHY6IjU5NDcifSx7bjoi5aSN5LuH6ICF6IGU55ufIix2OiI2MTIxIn0se246Iumck+iZuea3sea4iiIsdjoiNTc0MyJ9XX1dLDM6W3trZXk6ImNhdGVJZCIsbmFtZToi5YiG57G7Iix2YWx1ZTpbe246IueOi+iAheiNo+iAgCIsdjoiMjMzNiJ9LHtuOiLlkozlubPnsr7oi7EiLHY6IjMyMDMifSx7bjoi6Iux6ZuE6IGU55uf5omL5ri4Iix2OiI2MjAzIn0se246IkNG5omL5ri4Iix2OiIyNDEzIn0se246IumHkemTsumTsuS5i+aImCIsdjoiNzE4NSJ9LHtuOiLkuInlm73mnYAiLHY6IjE2NjkifSx7bjoi5Y6f56WeIix2OiI1NDg5In0se246Iuaji+eJjOahjOa4uCIsdjoiMTAwMDM2In0se246Iue7vOWQiOaJi+a4uCIsdjoiMTAwMDI5In0se246IuWKsuS5kOW5u+aDsyIsdjoiNzY5MSJ9LHtuOiLmlrDmuLjlub/lnLoiLHY6IjEwMDA1MiJ9LHtuOiLltKnlnY/vvJrmmJ/nqbnpk4HpgZMiLHY6IjczNDkifSx7bjoi54Gr5b2x5b+N6ICF5omL5ri4Iix2OiIyNDI5In0se246IuesrOS6lOS6uuagvCIsdjoiMzExNSJ9LHtuOiLpl67pgZPmiYvmuLgiLHY6IjI0NzcifSx7bjoi5pqX5Yy656qB5Zu0Iix2OiI3MjA5In0se246IlFR6aOe6L2m5omL5ri4Iix2OiIyOTI4In0se246IueQg+eQg+Wkp+S9nOaImCIsdjoiMjQxMSJ9LHtuOiLmmI7ml6XkuYvlkI4iLHY6IjM0ODMifSx7bjoi55qH5a6k5oiY5LqJIix2OiIyNDM5In0se246IkNPROaJi+a4uCIsdjoiNDc2OSJ9LHtuOiLmiYvmuLjkvJHpl7IiLHY6IjEwMDAwNCJ9LHtuOiLkuozmrKHlhYPmiYvmuLgiLHY6IjEwMDA5MSJ9LHtuOiLmkbjkuobkuKrpsbwiLHY6IjkyODMifSx7bjoiTU1PUlBHIix2OiIxMDAyNzMifSx7bjoi5Yqo5L2c5ri45oiPIix2OiIxMDAxOTcifSx7bjoi5oiY5LqJ5Yay56qBIix2OiI3NDQ5In0se246IueOi+iAheaooeaLn+aImCIsdjoiNTY5OSJ9LHtuOiLpgIbmsLTlr5LmiYvmuLgiLHY6Ijc3MjUifSx7bjoi5bm75aGUIix2OiI2NDM3In0se246IuasouS5kOaWl+WcsOS4uyIsdjoiMTc0OSJ9LHtuOiLnpZ7mraY05omL5ri4Iix2OiIzMTM1In0se246IumYtOmYs+W4iCIsdjoiMjU5OCJ9LHtuOiJETkbmiYvmuLgiLHY6IjQ5MjEifSx7bjoi5qyi5LmQ6bq75bCGIix2OiIxNzUxIn0se246IuiLsembhOiBlOebn+eUteernue7j+eQhiIsdjoiNzE3NyJ9LHtuOiLni7zkurrmnYDmiYvmuLgiLHY6IjEwMDA0OSJ9LHtuOiLmlrDlpKnpvpnlhavpg6jmiYvmuLgiLHY6IjY5NDUifSx7bjoi5Lit5Zu96LGh5qOLIix2OiIxNjcxIn0se246IuWkqeWkqeixoeajiyIsdjoiNDk5NyJ9LHtuOiLkvKDlpYfkuJbnlYwzRCIsdjoiMzk2MSJ9LHtuOiLlvqHpvpnlnKjlpKnmiYvmuLgiLHY6IjI1NjgifSx7bjoi6auY6IO96Iux6ZuEIix2OiI4MzU5In0se246IuWkqem+meWFq+mDqOaJi+a4uCIsdjoiMjg1MiJ9LHtuOiLkuInlm73lv5fmiJjnlaXniYgiLHY6IjU2MTkifSx7bjoi5b+N6ICF5b+F6aG75q27MyIsdjoiNDA0MSJ9LHtuOiJTS1nlhYnpgYciLHY6IjM3MTkifSx7bjoi6JmO54mZ6aKG5Li75LqJ6Zy4Iix2OiI3NTI5In0se246IuS6keS4iuWfjuS5i+atjCIsdjoiNTk3NyJ9LHtuOiLmmbbmoLgiLHY6IjcyNzkifSx7bjoi5LuZ5aKD5Lyg6K+0Uk8iLHY6IjI2NzUifSx7bjoi5aSp5aSp54u85Lq6Iix2OiIyNzc0In0se246IkpK5qOL54mMIix2OiIzODQxIn0se246IuWlh+i/uU1V77ya6KeJ6YaSIix2OiIzMTE2In0se246IuaMh+WwluWbm+W3nem6u+WwhiIsdjoiNzIxNSJ9LHtuOiLpg6jokL3lhrLnqoEiLHY6IjE3OTcifSx7bjoi5paw5YmR5L6g5oOF57yY5omL5ri4Iix2OiI2MjU5In0se246IuiQjOWuoOi/nOW+gSIsdjoiOTM4NSJ9LHtuOiLmmpfpu5HnoLTlnY/npZ7vvJrkuI3mnL0iLHY6IjYzODUifSx7bjoi6Iux6ZuE5p2AIix2OiIyNjg4In0se246IueDreihgOaxn+a5luaJi+a4uCIsdjoiMjgxNyJ9LHtuOiLom4vku5TmtL7lr7kiLHY6IjY5MDkifSx7bjoi5a6M576O5LiW55WM5omL5ri4Iix2OiI0MjM3In0se246IuW3heWzsOaImOiIsCIsdjoiMjUwMiJ9LHtuOiLni7zkurrmnYAiLHY6IjI3ODUifSx7bjoi5paX572X5aSn6ZmG77ya6a2C5biI5a+55YazIix2OiI2NzQ1In0se246IuatpuS+oOS5guaJi+a4uCIsdjoiNDkyOSJ9LHtuOiLor5vku5nmiYvmuLgiLHY6IjI2NDcifSx7bjoi5paX56C06IuN56m55omL5ri4Iix2OiI0MzM3In0se246Iui/t+S9oOS4lueVjCIsdjoiMjY4MyJ9LHtuOiLpo47kupEiLHY6IjMwNjEifSx7bjoi5aaE5oOz5bGx5rW3Iix2OiI2MDA3In0se246IuabmeWFieiLsembhCIsdjoiNjE2OSJ9LHtuOiLkuInlm73miJjnuqoyIix2OiI2MDQ5In0se246IuaipuW5u+aWsOivm+S7mSIsdjoiNTk3NSJ9LHtuOiLpu47mmI7op4nphpLvvJrnlJ/mnLoiLHY6IjYxMzEifSx7bjoi6L+c5b6BIix2OiI3ODM3In0se246IuiIqua1t+eOi++8mueHg+eDp+aEj+W/lyIsdjoiMzk0MyJ9LHtuOiLngavngqzkuYvlhYnvvJrml6DpmZAiLHY6IjYzOTkifSx7bjoi5beF5bOw5p6B6YCfIix2OiI2OTc5In0se246IuWco+Wig+S5i+WhlCIsdjoiNzA1NSJ9LHtuOiLprZTlipvlrp3otJ0iLHY6IjI4OTEifSx7bjoi6aaZ6IKg5rS+5a+5Iix2OiIzNjM5In0se246IuWIm+mAoOS4jumtlOazlSIsdjoiMjkzMSJ9LHtuOiJKSuaWl+WcsOS4uyIsdjoiNjI3MSJ9LHtuOiLmsLjmgZLnuqrlhYPvvJrmiJIiLHY6IjI2NDYifSx7bjoi5aSp5rav5piO5pyI5YiA5omL5ri4Iix2OiI1MTE1In0se246IueLvOS6uuadgOWumOaWuSIsdjoiMzY3OSJ9LHtuOiLmt7fmsozotbfmupAiLHY6IjU5ODUifSx7bjoi5aSa5aSa6Ieq6LWw5qOLIix2OiI1MTMzIn0se246IuaipuW5u+ivm+S7meaJi+a4uCIsdjoiMjY3MiJ9LHtuOiLpuL/lm77kuYvkuIsiLHY6IjYwMjcifSx7bjoi5paw56yR5YKy5rGf5rmWIix2OiI1NjY5In0se246IuWHoeS6uuS/ruS7meS8oO+8muS6uueVjOevhyIsdjoiODI5NyJ9LHtuOiLlpJrkuZDmo4vniYwiLHY6IjYyMDkifSx7bjoi5Y+j6KKL6KeJ6YaSIix2OiI1OTUzIn0se246Iui3kei3keWNoeS4gei9puaJi+a4uCIsdjoiMjYyMCJ9LHtuOiLlpbblnZciLHY6IjI3NzUifSx7bjoi5pyI5ZyG5LmL5aScIix2OiI0MzM5In0se246IueOh+Wcn+S5i+a7qCIsdjoiMjY5MSJ9LHtuOiLlvoHpgJQy5omL5ri4Iix2OiIyODExIn0se246IuiLsemtguS5i+WIg+WPo+iii+eJiCIsdjoiMjc2MCJ9LHtuOiLnsr7ngbXnm5vlhbjvvJrpu47mmI4iLHY6IjYxMjMifSx7bjoi5pa56Iif5omL5ri4Iix2OiI0MDM1In0se246IuaOvOibiyIsdjoiNjIyNSJ9LHtuOiLnu53kuJbku5nnjosiLHY6IjY2MTkifSx7bjoi5rWB5pif576k5L6g5LygIix2OiIzOTI3In0se246IuWvu+S7meaJi+a4uCIsdjoiMjk3OSJ9LHtuOiLkuIDmoqbmsZ/muZYiLHY6IjMwODIifSx7bjoiTG9zdCBMaWdodO+8iOiQpOeBq+eqgeWHu+WbvemZheacje+8iSIsdjoiNjg1OSJ9LHtuOiLlvLnluZXkupHmuLjmiI8iLHY6IjcwMDEifSx7bjoi54yO6a2C6KeJ6YaSIix2OiIzMDcxIn0se246IuWGkumZqeWym++8muaeq+S5i+S8oOivtCIsdjoiODAwNSJ9LHtuOiLlvoHpgJTmiYvmuLgiLHY6IjI1NTYifSx7bjoi5rW35bKb5aWH5YW1Iix2OiIyNjI0In0se246IuWAqeWls+W5vemtguaJi+a4uCIsdjoiMjUwMyJ9LHtuOiLotoXlh6HlhYjplIsiLHY6IjY1MDcifSx7bjoi6b6Z5LmL6LC3MuaJi+a4uCIsdjoiMjczNiJ9LHtuOiLltKnlnY8zIix2OiIyNjM5In0se246IueMq+WSjOiAgem8oCIsdjoiMjc1OCJ9LHtuOiLkuIPkurrkvKDlpYfvvJrlhYnkuI7mmpfkuYvkuqTmiJgiLHY6IjgxMjUifSx7bjoiSkrpurvlsIYiLHY6Ijk0ODcifSx7bjoi5ouJ6L2w6KW/5ri4Iix2OiI5NTQzIn0se246IuS4ieWbveaImOe6qiIsdjoiNjA0NyJ9LHtuOiLoh6rnlLHlubvmg7PmiYvmuLgiLHY6IjQwMTUifSx7bjoi56em5pe25piO5pyI5LiW55WMIix2OiI1Mjc5In0se246IuaWsOaWl+e9l+Wkp+mZhiIsdjoiNjY1NyJ9LHtuOiLmlrDnpZ7prZTlpKfpmYYiLHY6IjU5MzkifSx7bjoi6YCD6LeR5ZCn77yB5bCR5bm0Iix2OiI0MTM3In0se246IuWkquWPpOelnueOizIiLHY6IjY2NDkifSx7bjoi5YmR5L6g5LiW55WMMyIsdjoiNzE4MyJ9LHtuOiLlpKnlpKnlkIPpuKHmiYvmnLrniYgiLHY6IjQzNDEifSx7bjoi5pe256m654yO5Lq6MyIsdjoiNjQxMSJ9LHtuOiLlkIjph5HlvLnlpLTvvJrop4nphpIiLHY6IjY5MzEifSx7bjoi5piO5pel5pa56IifIix2OiI0OTI1In0se246IuWOn+Wni+W+gemAlCIsdjoiNzcxMyJ9LHtuOiLlpYfov7nvvJrmnIDlvLrogIUiLHY6IjMyMTUifSx7bjoi5aSp5aSp6YW36LeRIix2OiIxNzE1In0se246IkZDIOi2s+eQg+S4lueVjCIsdjoiMzg3MyJ9LHtuOiLkuIflm73op4nphpIiLHY6IjYxNTkifSx7bjoi5py65Yqo6YO95biC6Zi/5bCU5rOVIix2OiI1NDExIn0se246IuiIqua1t+eOi+eDreihgOiIque6vyIsdjoiNjE4MSJ9LHtuOiLlubvkuJbkuZ3mrYwiLHY6IjcxOTkifSx7bjoi5qSN54mp5aSn5oiY5YO15bC4Iix2OiI0ODUifSx7bjoi5peg5oKU5Y2O5aSPIix2OiI3MDYzIn0se246IuaXtuepuueMjuS6uiIsdjoiMTc0MiJ9LHtuOiLojZLph47kubHmlpciLHY6IjQ2MTMifSx7bjoi5ouz55qHOTjnu4jmnoHkuYvmiJhPTCIsdjoiMjY4NyJ9LHtuOiLom4fom4fkuonpnLgiLHY6IjI2ODAifSx7bjoi546L54mM56ue6YCfIix2OiI2NDYzIn0se246IumHjei/lOW4neWbvSIsdjoiNjk1NSJ9LHtuOiLlkJ7lmazmmJ/nqbrvvJrpu47mmI4iLHY6IjY2NTEifSx7bjoi5Y+j6KKL5aaW5oCqIix2OiIyNTQxIn0se246IlFR54Kr6Iie5omL5ri4Iix2OiIyOTkxIn0se246IuS4gOaLs+i2heS6uu+8muacgOW8uuS5i+eUtyIsdjoiNDYyOSJ9LHtuOiLojaPogIDmlrDkuInlm70iLHY6IjY5NDMifSx7bjoi5bCR5bm05LiJ5Zu95b+XMiIsdjoiNjEyNSJ9LHtuOiLmiJHnmoTotbfmupAiLHY6IjUzNjUifSx7bjoi5Yaz5oiY5bmz5a6J5LqsIix2OiIzMDY0In0se246IuWJkeeBtTIiLHY6IjcyMjMifSx7bjoi5byA5b+D5raI5raI5LmQIix2OiIxNzEyIn0se246IuWwj+Wwj+iageWbvSIsdjoiNzgwMyJ9LHtuOiLmnIDlvLpOQkEiLHY6IjI5ODgifSx7bjoi5YmR5L6g5oOF57yY5omL5ri4Iix2OiIyNjIxIn0se246IumVv+WuieW5u+aDsyIsdjoiNjcyNyJ9LHtuOiLmiJHlj6tNVDQiLHY6IjQwODcifSx7bjoi5YWo5piO5pif6KGX55CD5rS+5a+5Iix2OiI4NDAxIn0se246IuWkp+ivneilv+a4uOaJi+a4uCIsdjoiMjYyNiJ9LHtuOiLojaPogIDlpKflpKnkvb8iLHY6IjY0NzcifSx7bjoi6ZWH6a2C6KGX77ya5aSp55Sf5Li6546LIix2OiI2NTU3In0se246IuaRqeWwlOW6hOWbrSIsdjoiNTk4MSJ9LHtuOiLmuLjmiI/njovvvJrlhrPmlpfpk77mjqUiLHY6IjQ0NTEifSx7bjoi5YmR5L6g5LiW55WMMuaJi+a4uCIsdjoiMzE1MCJ9LHtuOiLpnZLkupHor4AyIix2OiI2MDA5In0se246IuaImOWcsOaXoOeWhiIsdjoiNzkwOSJ9LHtuOiLkuIDlv7XpgI3pgaUiLHY6IjY0MTkifSx7bjoi5rC45Yqr5peg6Ze05omL5ri4Iix2OiI3NTc5In0se246IuWwmOeZveemgeWMuiIsdjoiNzI5NyJ9LHtuOiLlhYPmoqbkuYvmmJ8iLHY6Ijk1MjEifSx7bjoi5LiN6Imv5Lq6MyIsdjoiNTg5MSJ9LHtuOiLliZHngbXvvJrpnanlkb0iLHY6IjQ1NDUifSx7bjoi6a2U5Yqb5a6d6LSd77ya5peF5Lq6Iix2OiI3NTczIn0se246Iuexs+WKoOWwj+mVhyIsdjoiNzI2OSJ9LHtuOiLpvpnmrabmiYvmuLgiLHY6IjUyMTkifSx7bjoi5paX572X5aSn6ZmGMu+8mue7neS4luWUkOmXqCIsdjoiNjU4MSJ9LHtuOiLopb/ooYznuqrnh4PprYIiLHY6IjgzMDMifSx7bjoi5Z2m5YWL5LiW55WM6Zeq5Ye75oiYIix2OiI0OTc3In0se246IuWGm+ajiyIsdjoiMjU2MSJ9LHtuOiLppaXojZLvvJrmlrDlrrYiLHY6IjY0OTEifSx7bjoi5ouz55qH5ZG96L+QIix2OiIzMzc5In0se246IuWunuWGtei2s+eQgyIsdjoiMzc0MSJ9LHtuOiLmiJjoiLDkuJbnlYzpl6rlh7vmiJgiLHY6IjQxMDEifSx7bjoi5pe256m65Y+s5ZSkIix2OiIyNTUxIn0se246IueOi+eJjOaImOS6ie+8muaWh+aYjumHjeWQryIsdjoiNTQ3OSJ9LHtuOiLpm4DprYLpurvlsIYiLHY6IjcxMDcifSx7bjoi5qyi5LmQ5Y2H57qnIix2OiIzOTI1In0se246Iue7v+iJsuW+gemAlCIsdjoiNDIyNyJ9LHtuOiLlvLnlvLnloILmiYvmuLgiLHY6IjI4NTcifSx7bjoi5aSq5p6B54aK54yrM++8mueMjum+mSIsdjoiMjc3OCJ9LHtuOiLlk4jliKnms6LnibnvvJrprZTms5Xop4nphpIiLHY6IjU4MzUifSx7bjoi5aSp5Zyw5Yqr77ya5bm95Z+O5YaN5Li0Iix2OiI1OTg3In0se246IueDreihgOihl+evriIsdjoiNTg1OSJ9LHtuOiLnpZ7pm5XkvqDkvqPmiYvmuLgiLHY6IjE3ODEifSx7bjoi5bGx5rW36ZWc6IqxIix2OiI1MDg5In0se246IuS4ieWbveW/l+aImOaji+eJiCIsdjoiNzkzNyJ9LHtuOiLnpZ7pm5XkvqDkvqMyIix2OiI0MjA5In0se246IuS7memtlOWGsyIsdjoiMTY3NCJ9LHtuOiLnjovogIXojaPogIDmmJ/kuYvnoLTmmZMiLHY6Ijc5MjcifSx7bjoi5oiR5Zyo5rGf5rmW5LmL56We6a2U6YGTIix2OiI3Njk5In0se246IuaipuW5u+aooeaLn+aImCIsdjoiMzQ4MSJ9LHtuOiLljZXmnLrmiYvmuLgiLHY6IjI3NzcifSx7bjoi5paX572X5aSn6ZmGLeaWl+elnuWGjeS4tCIsdjoiNjYzMSJ9LHtuOiLmnKrmnaXkuYvlvbkiLHY6IjY4MzEifSx7bjoi6aOO5LqR5bKb6KGM5YqoIix2OiI0Njk1In0se246IuaWsOa4uOaOqOiNkCIsdjoiMzE2MCJ9LHtuOiLngavlvbHlv43ogIVPTCIsdjoiMzkwMSJ9LHtuOiLkuZ3ngbXnpZ7ln58iLHY6Ijc3MTkifSx7bjoi5q2m5Yqo5Lm+5Z2kIix2OiIzODI5In0se246IuenpuaXtuaYjuaciDIiLHY6IjE3ODQifSx7bjoi5paH5piO5LiO5b6B5pyNIix2OiI3MDcxIn0se246IuaImOWPjO+8muW4leW8peS7gCIsdjoiNDEzMyJ9LHtuOiLlpKfoiKrmtbfml7bku6PvvJrmtbfkuIrpnLjkuLsiLHY6IjY5MjkifSx7bjoi5YmR572RMe+8muW9kuadpSIsdjoiNzM2MSJ9LHtuOiLnu53ljLrpm7YiLHY6Ijc3MTEifSx7bjoi6buR6Imy5rKZ5ryg5omL5ri4Iix2OiI3Mjg3In0se246IuiZjueJmeWQg+m4oSIsdjoiNzQ2NSJ9LHtuOiLkuIDliZHmlqnku5kiLHY6IjY4NDMifSx7bjoi5Lyg5aWH5aSp5LiLIix2OiI2OTI3In0se246IuaWl+aWl+WggiIsdjoiNzEzMyJ9LHtuOiLmlpfnvZflpKfpmYYiLHY6IjYxMTkifSx7bjoi5aSp6LCV5omL5ri4Iix2OiI1OTI1In0se246IuWdjuWFrOmqkeWGoOWJkSIsdjoiNjY0MSJ9LHtuOiLmnIDnu4jlubvmg7Pop4nphpIiLHY6IjI3MjEifSx7bjoi56We5bCG5LiJ5Zu9Iix2OiI2NjIxIn0se246IueBjOevrumrmOaJi+ato+eJiOaOiOadg+aJi+a4uCIsdjoiNTM5OSJ9LHtuOiLliZHkuI7lrrblm60iLHY6IjI4MzgifSx7bjoi5p6B5peg5Y+MMiIsdjoiNzgyNSJ9LHtuOiLlhYnmmI7lpKfpmYYiLHY6IjI4MzIifSx7bjoi6I2S6YeO6KGM5YqoIix2OiIzMDg0In0se246IuaImOaWl+azleWImSIsdjoiOTUxMyJ9LHtuOiLnlq/ni4Lljp/lp4vkuroiLHY6IjQ2MTkifSx7bjoi6YCG5oiY5omL5ri4Iix2OiI3NTc1In0se246Iuefs+WZqOaXtuS7o++8muiniemGkiIsdjoiOTE1OSJ9XX1dfSxzZWFyY2hVcmw6Imh0dHBzOi8vc2VhcmNoLmNkbi5odXlhLmNvbS8/bT1TZWFyY2gmZG89Z2V0U2VhcmNoQ29udGVudCZxPSoqJnVpZD0wJnY9NCZ0eXA9LTUmbGl2ZXN0YXRlPTAmcm93cz00MCZzdGFydD0wIixzZWFyY2hhYmxlOjIscXVpY2tTZWFyY2g6MCxoZWFkZXJzOnsiVXNlci1BZ2VudCI6Ik1PQklMRV9VQSJ9LHRpbWVvdXQ6NWUzLGxpbWl0OjgscGxheV9wYXJzZTohMCxsYXp5OiJqczpcbiAgICAgICAgbGV0IHJpZCA9IGlucHV0Lm1hdGNoKC9cXC8gKFxcZCArICkgLyApWzFdO1xuICAgICAgICBmdW5jdGlvbiBnZXRSZWFsVXJsKGxpdmVfdXJsKSB7XG4gICAgICAgICAgICBsZXQgW2ksIGJdID0gbGl2ZV91cmwuc3BsaXQoJz8nKTtcbiAgICAgICAgICAgIGxldCByID0gaS5zcGxpdCgnLycpLnBvcCgpO1xuICAgICAgICAgICAgbGV0IHMgPSByLnJlcGxhY2UoLy4oZmx2fG0zdTgpLywgJycpO1xuICAgICAgICAgICAgbGV0IGNfdG1wID0gYi5zcGxpdCgnJicpLmZpbHRlcihuID0+IG4pO1xuICAgICAgICAgICAgbGV0IG4gPSB7fTtcbiAgICAgICAgICAgIGxldCBjX3RtcDIgPSBbXTtcbiAgICAgICAgICAgIGNfdG1wLmZvckVhY2goZnVuY3Rpb24odG1wLCBpbmRleCkge1xuICAgICAgICAgICAgICAgIGlmIChpbmRleCA8IDMpIHtcbiAgICAgICAgICAgICAgICAgICAgblt0bXAuc3BsaXQoJz0nKVswXV0gPSB0bXAuc3BsaXQoJz0nKVsxXVxuICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAgIGNfdG1wMi5wdXNoKHRtcClcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIGxldCB0bXAyID0gY190bXAyLmpvaW4oJyYnKTtcbiAgICAgICAgICAgIG5bdG1wMi5zcGxpdCgnPScpWzBdXSA9IHRtcDIuc3BsaXQoJz0nKVsxXTtcbiAgICAgICAgICAgIGxldCBmbSA9IGRlY29kZVVSSUNvbXBvbmVudChuLmZtKS5zcGxpdCgnJicpWzBdO1xuICAgICAgICAgICAgbGV0IHUgPSBiYXNlNjREZWNvZGUoZm0pO1xuICAgICAgICAgICAgbGV0IHAgPSB1LnNwbGl0KCdfJylbMF07XG4gICAgICAgICAgICBsZXQgZiA9IG5ldyBEYXRlKCkuZ2V0VGltZSgpICsgJzAwMDAnO1xuICAgICAgICAgICAgbGV0IGxsID0gbi53c1RpbWU7XG4gICAgICAgICAgICBsZXQgdCA9ICcwJztcbiAgICAgICAgICAgIGxldCBoID0gW3AsIHQsIHMsIGYsIGxsXS5qb2luKCdfJyk7XG4gICAgICAgICAgICBsZXQgbSA9IG1kNShoKTtcbiAgICAgICAgICAgIHJldHVybiAoaSArICc/d3NTZWNyZXQ9JyArIG0gKyAnJndzVGltZT0nICsgbGwgKyAnJnU9JyArIHQgKyAnJnNlcWlkPScgKyBmICsgJyYnICsgY190bXAyLnBvcCgpKS5yZXBsYWNlKCdobHMnLCAnZmx2JykucmVwbGFjZSgnbTN1OCcsICdmbHYnKVxuICAgICAgICB9XG4gICAgICAgIGxldCBwdXJsID0gSlNPTi5wYXJzZShyZXF1ZXN0KCdodHRwczovL21wLmh1eWEuY29tL2NhY2hlLnBocD9tPUxpdmUmZG89cHJvZmlsZVJvb20mcm9vbWlkPScgKyByaWQpKS5kYXRhLnN0cmVhbS5mbHYubXVsdGlMaW5lWzBdLnVybDtcbiAgICAgICAgaW5wdXQgPSB7XG4gICAgICAgICAgICBqeDogMCxcbiAgICAgICAgICAgIHVybDogZ2V0UmVhbFVybChwdXJsKSxcbiAgICAgICAgICAgIHBhcnNlOiAwLFxuICAgICAgICAgICAgaGVhZGVyOiBKU09OLnN0cmluZ2lmeSh7XG4gICAgICAgICAgICAgICAgJ3VzZXItYWdlbnQnOiAnTW96aWxsYS81LjAnXG4gICAgICAgICAgICB9KVxuICAgICAgICB9XG4gICAgIiwi5o6o6I2QIjoianM6XG4gICAgICAgIGxldCBkID0gW107XG4gICAgICAgIGxldCBqbyA9IEpTT04ucGFyc2UocmVxdWVzdChpbnB1dCkpLmRhdGEuZGF0YXM7XG4gICAgICAgIGpvLmZvckVhY2goaXQgPT4ge1xuICAgICAgICAgICAgICAgIGQucHVzaCh7XG4gICAgICAgICAgICAgICAgICAgIHVybDogaXQucHJvZmlsZVJvb20sXG4gICAgICAgICAgICAgICAgICAgIHRpdGxlOiBpdC5pbnRyb2R1Y3Rpb24sXG4gICAgICAgICAgICAgICAgICAgIGltZzogaXQuc2NyZWVuc2hvdCxcbiAgICAgICAgICAgICAgICAgICAgZGVzYzogJ/CfkYEnICsgaXQudG90YWxDb3VudCArICcgIPCfhpknICsgaXQubmljayxcbiAgICAgICAgICAgICAgICB9KVxuICAgICAgICB9KTtcbiAgICAgICAgc2V0UmVzdWx0KGQpO1xuICAgICIsIuS4gOe6pyI6ImpzOlxuICAgICAgICBsZXQgZCA9IFtdO1xuICAgICAgICBsZXQgam8gPSBKU09OLnBhcnNlKHJlcXVlc3QoaW5wdXQpKS5kYXRhLmRhdGFzO1xuICAgICAgICBqby5mb3JFYWNoKGl0ID0+IHtcbiAgICAgICAgICAgICAgICBkLnB1c2goe1xuICAgICAgICAgICAgICAgICAgICB1cmw6IGl0LnByb2ZpbGVSb29tLFxuICAgICAgICAgICAgICAgICAgICB0aXRsZTogaXQuaW50cm9kdWN0aW9uLFxuICAgICAgICAgICAgICAgICAgICBpbWc6IGl0LnNjcmVlbnNob3QsXG4gICAgICAgICAgICAgICAgICAgIGRlc2M6ICfwn5GBJyArIGl0LnRvdGFsQ291bnQgKyAnICDwn4aZJyArIGl0Lm5pY2ssXG4gICAgICAgICAgICAgICAgfSlcbiAgICAgICAgfSk7XG4gICAgICAgIHNldFJlc3VsdChkKTtcbiAgICAiLCLkuoznuqciOiIqIiwi5pCc57SiIjoianNvbjpyZXNwb25zZS4zLmRvY3M7Z2FtZV9yb29tTmFtZTtnYW1lX3NjcmVlbnNob3Q7Z2FtZV9uaWNrO3Jvb21faWQifTs= \ No newline at end of file diff --git a/jtxtv10/lib/ip.json b/jtxtv10/lib/ip.json new file mode 100644 index 0000000..37b2e41 --- /dev/null +++ b/jtxtv10/lib/ip.json @@ -0,0 +1 @@ +https://test.cms.bidiys.com/ \ No newline at end of file diff --git a/jtxtv10/lib/jinja.js b/jtxtv10/lib/jinja.js new file mode 100644 index 0000000..80737af --- /dev/null +++ b/jtxtv10/lib/jinja.js @@ -0,0 +1,577 @@ +/*! + * Jinja Templating for JavaScript v0.1.8 + * https://github.com/sstur/jinja-js + * + * This is a slimmed-down Jinja2 implementation [http://jinja.pocoo.org/] + * + * In the interest of simplicity, it deviates from Jinja2 as follows: + * - Line statements, cycle, super, macro tags and block nesting are not implemented + * - auto escapes html by default (the filter is "html" not "e") + * - Only "html" and "safe" filters are built in + * - Filters are not valid in expressions; `foo|length > 1` is not valid + * - Expression Tests (`if num is odd`) not implemented (`is` translates to `==` and `isnot` to `!=`) + * + * Notes: + * - if property is not found, but method '_get' exists, it will be called with the property name (and cached) + * - `{% for n in obj %}` iterates the object's keys; get the value with `{% for n in obj %}{{ obj[n] }}{% endfor %}` + * - subscript notation `a[0]` takes literals or simple variables but not `a[item.key]` + * - `.2` is not a valid number literal; use `0.2` + * + */ +/*global require, exports, module, define */ + +(function (global, factory) { + typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports) : + typeof define === 'function' && define.amd ? define(['exports'], factory) : + (global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global.jinja = {})); +})(this, (function (jinja) { + "use strict"; + var STRINGS = /'(\\.|[^'])*'|"(\\.|[^"'"])*"/g; + var IDENTS_AND_NUMS = /([$_a-z][$\w]*)|([+-]?\d+(\.\d+)?)/g; + var NUMBER = /^[+-]?\d+(\.\d+)?$/; + //non-primitive literals (array and object literals) + var NON_PRIMITIVES = /\[[@#~](,[@#~])*\]|\[\]|\{([@i]:[@#~])(,[@i]:[@#~])*\}|\{\}/g; + //bare identifiers such as variables and in object literals: {foo: 'value'} + var IDENTIFIERS = /[$_a-z][$\w]*/ig; + var VARIABLES = /i(\.i|\[[@#i]\])*/g; + var ACCESSOR = /(\.i|\[[@#i]\])/g; + var OPERATORS = /(===?|!==?|>=?|<=?|&&|\|\||[+\-\*\/%])/g; + //extended (english) operators + var EOPS = /(^|[^$\w])(and|or|not|is|isnot)([^$\w]|$)/g; + var LEADING_SPACE = /^\s+/; + var TRAILING_SPACE = /\s+$/; + + var START_TOKEN = /\{\{\{|\{\{|\{%|\{#/; + var TAGS = { + '{{{': /^('(\\.|[^'])*'|"(\\.|[^"'"])*"|.)+?\}\}\}/, + '{{': /^('(\\.|[^'])*'|"(\\.|[^"'"])*"|.)+?\}\}/, + '{%': /^('(\\.|[^'])*'|"(\\.|[^"'"])*"|.)+?%\}/, + '{#': /^('(\\.|[^'])*'|"(\\.|[^"'"])*"|.)+?#\}/ + }; + + var delimeters = { + '{%': 'directive', + '{{': 'output', + '{#': 'comment' + }; + + var operators = { + and: '&&', + or: '||', + not: '!', + is: '==', + isnot: '!=' + }; + + var constants = { + 'true': true, + 'false': false, + 'null': null + }; + + function Parser() { + this.nest = []; + this.compiled = []; + this.childBlocks = 0; + this.parentBlocks = 0; + this.isSilent = false; + } + + Parser.prototype.push = function (line) { + if (!this.isSilent) { + this.compiled.push(line); + } + }; + + Parser.prototype.parse = function (src) { + this.tokenize(src); + return this.compiled; + }; + + Parser.prototype.tokenize = function (src) { + var lastEnd = 0, parser = this, trimLeading = false; + matchAll(src, START_TOKEN, function (open, index, src) { + //here we match the rest of the src against a regex for this tag + var match = src.slice(index + open.length).match(TAGS[open]); + match = (match ? match[0] : ''); + //here we sub out strings so we don't get false matches + var simplified = match.replace(STRINGS, '@'); + //if we don't have a close tag or there is a nested open tag + if (!match || ~simplified.indexOf(open)) { + return index + 1; + } + var inner = match.slice(0, 0 - open.length); + //check for white-space collapse syntax + if (inner.charAt(0) === '-') var wsCollapseLeft = true; + if (inner.slice(-1) === '-') var wsCollapseRight = true; + inner = inner.replace(/^-|-$/g, '').trim(); + //if we're in raw mode and we are not looking at an "endraw" tag, move along + if (parser.rawMode && (open + inner) !== '{%endraw') { + return index + 1; + } + var text = src.slice(lastEnd, index); + lastEnd = index + open.length + match.length; + if (trimLeading) text = trimLeft(text); + if (wsCollapseLeft) text = trimRight(text); + if (wsCollapseRight) trimLeading = true; + if (open === '{{{') { + //liquid-style: make {{{x}}} => {{x|safe}} + open = '{{'; + inner += '|safe'; + } + parser.textHandler(text); + parser.tokenHandler(open, inner); + }); + var text = src.slice(lastEnd); + if (trimLeading) text = trimLeft(text); + this.textHandler(text); + }; + + Parser.prototype.textHandler = function (text) { + this.push('write(' + JSON.stringify(text) + ');'); + }; + + Parser.prototype.tokenHandler = function (open, inner) { + var type = delimeters[open]; + if (type === 'directive') { + this.compileTag(inner); + } else if (type === 'output') { + var extracted = this.extractEnt(inner, STRINGS, '@'); + //replace || operators with ~ + extracted.src = extracted.src.replace(/\|\|/g, '~').split('|'); + //put back || operators + extracted.src = extracted.src.map(function (part) { + return part.split('~').join('||'); + }); + var parts = this.injectEnt(extracted, '@'); + if (parts.length > 1) { + var filters = parts.slice(1).map(this.parseFilter.bind(this)); + this.push('filter(' + this.parseExpr(parts[0]) + ',' + filters.join(',') + ');'); + } else { + this.push('filter(' + this.parseExpr(parts[0]) + ');'); + } + } + }; + + Parser.prototype.compileTag = function (str) { + var directive = str.split(' ')[0]; + var handler = tagHandlers[directive]; + if (!handler) { + throw new Error('Invalid tag: ' + str); + } + handler.call(this, str.slice(directive.length).trim()); + }; + + Parser.prototype.parseFilter = function (src) { + src = src.trim(); + var match = src.match(/[:(]/); + var i = match ? match.index : -1; + if (i < 0) return JSON.stringify([src]); + var name = src.slice(0, i); + var args = src.charAt(i) === ':' ? src.slice(i + 1) : src.slice(i + 1, -1); + args = this.parseExpr(args, {terms: true}); + return '[' + JSON.stringify(name) + ',' + args + ']'; + }; + + Parser.prototype.extractEnt = function (src, regex, placeholder) { + var subs = [], isFunc = typeof placeholder == 'function'; + src = src.replace(regex, function (str) { + var replacement = isFunc ? placeholder(str) : placeholder; + if (replacement) { + subs.push(str); + return replacement; + } + return str; + }); + return {src: src, subs: subs}; + }; + + Parser.prototype.injectEnt = function (extracted, placeholder) { + var src = extracted.src, subs = extracted.subs, isArr = Array.isArray(src); + var arr = (isArr) ? src : [src]; + var re = new RegExp('[' + placeholder + ']', 'g'), i = 0; + arr.forEach(function (src, index) { + arr[index] = src.replace(re, function () { + return subs[i++]; + }); + }); + return isArr ? arr : arr[0]; + }; + + //replace complex literals without mistaking subscript notation with array literals + Parser.prototype.replaceComplex = function (s) { + var parsed = this.extractEnt(s, /i(\.i|\[[@#i]\])+/g, 'v'); + parsed.src = parsed.src.replace(NON_PRIMITIVES, '~'); + return this.injectEnt(parsed, 'v'); + }; + + //parse expression containing literals (including objects/arrays) and variables (including dot and subscript notation) + //valid expressions: `a + 1 > b.c or c == null`, `a and b[1] != c`, `(a < b) or (c < d and e)`, 'a || [1]` + Parser.prototype.parseExpr = function (src, opts) { + opts = opts || {}; + //extract string literals -> @ + var parsed1 = this.extractEnt(src, STRINGS, '@'); + //note: this will catch {not: 1} and a.is; could we replace temporarily and then check adjacent chars? + parsed1.src = parsed1.src.replace(EOPS, function (s, before, op, after) { + return (op in operators) ? before + operators[op] + after : s; + }); + //sub out non-string literals (numbers/true/false/null) -> # + // the distinction is necessary because @ can be object identifiers, # cannot + var parsed2 = this.extractEnt(parsed1.src, IDENTS_AND_NUMS, function (s) { + return (s in constants || NUMBER.test(s)) ? '#' : null; + }); + //sub out object/variable identifiers -> i + var parsed3 = this.extractEnt(parsed2.src, IDENTIFIERS, 'i'); + //remove white-space + parsed3.src = parsed3.src.replace(/\s+/g, ''); + + //the rest of this is simply to boil the expression down and check validity + var simplified = parsed3.src; + //sub out complex literals (objects/arrays) -> ~ + // the distinction is necessary because @ and # can be subscripts but ~ cannot + while (simplified !== (simplified = this.replaceComplex(simplified))) ; + //now @ represents strings, # represents other primitives and ~ represents non-primitives + //replace complex variables (those with dot/subscript accessors) -> v + while (simplified !== (simplified = simplified.replace(/i(\.i|\[[@#i]\])+/, 'v'))) ; + //empty subscript or complex variables in subscript, are not permitted + simplified = simplified.replace(/[iv]\[v?\]/g, 'x'); + //sub in "i" for @ and # and ~ and v (now "i" represents all literals, variables and identifiers) + simplified = simplified.replace(/[@#~v]/g, 'i'); + //sub out operators + simplified = simplified.replace(OPERATORS, '%'); + //allow 'not' unary operator + simplified = simplified.replace(/!+[i]/g, 'i'); + var terms = opts.terms ? simplified.split(',') : [simplified]; + terms.forEach(function (term) { + //simplify logical grouping + while (term !== (term = term.replace(/\(i(%i)*\)/g, 'i'))) ; + if (!term.match(/^i(%i)*/)) { + throw new Error('Invalid expression: ' + src + " " + term); + } + }); + parsed3.src = parsed3.src.replace(VARIABLES, this.parseVar.bind(this)); + parsed2.src = this.injectEnt(parsed3, 'i'); + parsed1.src = this.injectEnt(parsed2, '#'); + return this.injectEnt(parsed1, '@'); + }; + + Parser.prototype.parseVar = function (src) { + var args = Array.prototype.slice.call(arguments); + var str = args.pop(), index = args.pop(); + //quote bare object identifiers (might be a reserved word like {while: 1}) + if (src === 'i' && str.charAt(index + 1) === ':') { + return '"i"'; + } + var parts = ['"i"']; + src.replace(ACCESSOR, function (part) { + if (part === '.i') { + parts.push('"i"'); + } else if (part === '[i]') { + parts.push('get("i")'); + } else { + parts.push(part.slice(1, -1)); + } + }); + return 'get(' + parts.join(',') + ')'; + }; + + //escapes a name to be used as a javascript identifier + Parser.prototype.escName = function (str) { + return str.replace(/\W/g, function (s) { + return '$' + s.charCodeAt(0).toString(16); + }); + }; + + Parser.prototype.parseQuoted = function (str) { + if (str.charAt(0) === "'") { + str = str.slice(1, -1).replace(/\\.|"/, function (s) { + if (s === "\\'") return "'"; + return s.charAt(0) === '\\' ? s : ('\\' + s); + }); + str = '"' + str + '"'; + } + //todo: try/catch or deal with invalid characters (linebreaks, control characters) + return JSON.parse(str); + }; + + + //the context 'this' inside tagHandlers is the parser instance + var tagHandlers = { + 'if': function (expr) { + this.push('if (' + this.parseExpr(expr) + ') {'); + this.nest.unshift('if'); + }, + 'else': function () { + if (this.nest[0] === 'for') { + this.push('}, function() {'); + } else { + this.push('} else {'); + } + }, + 'elseif': function (expr) { + this.push('} else if (' + this.parseExpr(expr) + ') {'); + }, + 'endif': function () { + this.nest.shift(); + this.push('}'); + }, + 'for': function (str) { + var i = str.indexOf(' in '); + var name = str.slice(0, i).trim(); + var expr = str.slice(i + 4).trim(); + this.push('each(' + this.parseExpr(expr) + ',' + JSON.stringify(name) + ',function() {'); + this.nest.unshift('for'); + }, + 'endfor': function () { + this.nest.shift(); + this.push('});'); + }, + 'raw': function () { + this.rawMode = true; + }, + 'endraw': function () { + this.rawMode = false; + }, + 'set': function (stmt) { + var i = stmt.indexOf('='); + var name = stmt.slice(0, i).trim(); + var expr = stmt.slice(i + 1).trim(); + this.push('set(' + JSON.stringify(name) + ',' + this.parseExpr(expr) + ');'); + }, + 'block': function (name) { + if (this.isParent) { + ++this.parentBlocks; + var blockName = 'block_' + (this.escName(name) || this.parentBlocks); + this.push('block(typeof ' + blockName + ' == "function" ? ' + blockName + ' : function() {'); + } else if (this.hasParent) { + this.isSilent = false; + ++this.childBlocks; + blockName = 'block_' + (this.escName(name) || this.childBlocks); + this.push('function ' + blockName + '() {'); + } + this.nest.unshift('block'); + }, + 'endblock': function () { + this.nest.shift(); + if (this.isParent) { + this.push('});'); + } else if (this.hasParent) { + this.push('}'); + this.isSilent = true; + } + }, + 'extends': function (name) { + name = this.parseQuoted(name); + var parentSrc = this.readTemplateFile(name); + this.isParent = true; + this.tokenize(parentSrc); + this.isParent = false; + this.hasParent = true; + //silence output until we enter a child block + this.isSilent = true; + }, + 'include': function (name) { + name = this.parseQuoted(name); + var incSrc = this.readTemplateFile(name); + this.isInclude = true; + this.tokenize(incSrc); + this.isInclude = false; + } + }; + + //liquid style + tagHandlers.assign = tagHandlers.set; + //python/django style + tagHandlers.elif = tagHandlers.elseif; + + var getRuntime = function runtime(data, opts) { + var defaults = {autoEscape: 'toJson'}; + var _toString = Object.prototype.toString; + var _hasOwnProperty = Object.prototype.hasOwnProperty; + var getKeys = Object.keys || function (obj) { + var keys = []; + for (var n in obj) if (_hasOwnProperty.call(obj, n)) keys.push(n); + return keys; + }; + var isArray = Array.isArray || function (obj) { + return _toString.call(obj) === '[object Array]'; + }; + var create = Object.create || function (obj) { + function F() { + } + + F.prototype = obj; + return new F(); + }; + var toString = function (val) { + if (val == null) return ''; + return (typeof val.toString == 'function') ? val.toString() : _toString.call(val); + }; + var extend = function (dest, src) { + var keys = getKeys(src); + for (var i = 0, len = keys.length; i < len; i++) { + var key = keys[i]; + dest[key] = src[key]; + } + return dest; + }; + //get a value, lexically, starting in current context; a.b -> get("a","b") + var get = function () { + var val, n = arguments[0], c = stack.length; + while (c--) { + val = stack[c][n]; + if (typeof val != 'undefined') break; + } + for (var i = 1, len = arguments.length; i < len; i++) { + if (val == null) continue; + n = arguments[i]; + val = (_hasOwnProperty.call(val, n)) ? val[n] : (typeof val._get == 'function' ? (val[n] = val._get(n)) : null); + } + return (val == null) ? '' : val; + }; + var set = function (n, val) { + stack[stack.length - 1][n] = val; + }; + var push = function (ctx) { + stack.push(ctx || {}); + }; + var pop = function () { + stack.pop(); + }; + var write = function (str) { + output.push(str); + }; + var filter = function (val) { + for (var i = 1, len = arguments.length; i < len; i++) { + var arr = arguments[i], name = arr[0], filter = filters[name]; + if (filter) { + arr[0] = val; + //now arr looks like [val, arg1, arg2] + val = filter.apply(data, arr); + } else { + throw new Error('Invalid filter: ' + name); + } + } + if (opts.autoEscape && name !== opts.autoEscape && name !== 'safe') { + //auto escape if not explicitly safe or already escaped + val = filters[opts.autoEscape].call(data, val); + } + output.push(val); + }; + var each = function (obj, loopvar, fn1, fn2) { + if (obj == null) return; + var arr = isArray(obj) ? obj : getKeys(obj), len = arr.length; + var ctx = {loop: {length: len, first: arr[0], last: arr[len - 1]}}; + push(ctx); + for (var i = 0; i < len; i++) { + extend(ctx.loop, {index: i + 1, index0: i}); + fn1(ctx[loopvar] = arr[i]); + } + if (len === 0 && fn2) fn2(); + pop(); + }; + var block = function (fn) { + push(); + fn(); + pop(); + }; + var render = function () { + return output.join(''); + }; + data = data || {}; + opts = extend(defaults, opts || {}); + var filters = extend({ + html: function (val) { + return toString(val) + .split('&').join('&') + .split('<').join('<') + .split('>').join('>') + .split('"').join('"'); + }, + safe: function (val) { + return val; + }, + toJson: function (val) { + if (typeof val === 'object') { + return JSON.stringify(val); + } + return toString(val); + } + }, opts.filters || {}); + var stack = [create(data || {})], output = []; + return { + get: get, + set: set, + push: push, + pop: pop, + write: write, + filter: filter, + each: each, + block: block, + render: render + }; + }; + + var runtime; + + jinja.compile = function (markup, opts) { + opts = opts || {}; + var parser = new Parser(); + parser.readTemplateFile = this.readTemplateFile; + var code = []; + code.push('function render($) {'); + code.push('var get = $.get, set = $.set, push = $.push, pop = $.pop, write = $.write, filter = $.filter, each = $.each, block = $.block;'); + code.push.apply(code, parser.parse(markup)); + code.push('return $.render();'); + code.push('}'); + code = code.join('\n'); + if (opts.runtime === false) { + var fn = new Function('data', 'options', 'return (' + code + ')(runtime(data, options))'); + } else { + runtime = runtime || (runtime = getRuntime.toString()); + fn = new Function('data', 'options', 'return (' + code + ')((' + runtime + ')(data, options))'); + } + return {render: fn}; + }; + + jinja.render = function (markup, data, opts) { + var tmpl = jinja.compile(markup); + return tmpl.render(data, opts); + }; + + jinja.templateFiles = []; + + jinja.readTemplateFile = function (name) { + var templateFiles = this.templateFiles || []; + var templateFile = templateFiles[name]; + if (templateFile == null) { + throw new Error('Template file not found: ' + name); + } + return templateFile; + }; + + + /*! + * Helpers + */ + + function trimLeft(str) { + return str.replace(LEADING_SPACE, ''); + } + + function trimRight(str) { + return str.replace(TRAILING_SPACE, ''); + } + + function matchAll(str, reg, fn) { + //copy as global + reg = new RegExp(reg.source, 'g' + (reg.ignoreCase ? 'i' : '') + (reg.multiline ? 'm' : '')); + var match; + while ((match = reg.exec(str))) { + var result = fn(match[0], match.index, str); + if (typeof result == 'number') { + reg.lastIndex = result; + } + } + } +})); \ No newline at end of file