A website to compare travel times in Velib at Paris.

index.php 37KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397
  1. <?php
  2. session_start();
  3. date_default_timezone_set("Europe/Paris");
  4. function search($array) {
  5. global $_POST;
  6. return in_array($_POST['start_search'], $array) && in_array($_POST['end_search'], $array);
  7. }
  8. // Generates a token against CSRF
  9. // ==============================
  10. function generate_token($name = '') {
  11. if(session_id() == '')
  12. session_start();
  13. $token = uniqid(rand(), true);
  14. $_SESSION[$name.'_token'] = $token;
  15. $_SESSION[$name.'_token_time'] = time();
  16. return $token;
  17. }
  18. // Checks that the anti-CSRF token is correct
  19. // ==========================================
  20. function check_token($time, $name = '') {
  21. if(session_id() == '')
  22. session_start();
  23. if(isset($_SESSION[$name.'_token']) && isset($_SESSION[$name.'_token_time']) && (isset($_POST['token']) || isset($_GET['token']))) {
  24. if(!empty($_POST['token']))
  25. $token = $_POST['token'];
  26. else
  27. $token = $_GET['token'];
  28. if($_SESSION[$name.'_token'] == $token) {
  29. if($_SESSION[$name.'_token_time'] >= (time() - (int) $time))
  30. return true;
  31. }
  32. }
  33. return false;
  34. }
  35. if(is_file('data/data')) {
  36. $data = unserialize(gzinflate(base64_decode(file_get_contents('data/data'))));
  37. }
  38. else {
  39. $data = array();
  40. }
  41. if(!empty($_GET['suppr']) && !empty($_SESSION['admin']) && !empty($_GET['token'])) {
  42. if(check_token(600, 'defivelib')) {
  43. unlink($data[$_GET['suppr']]);
  44. }
  45. else {
  46. exit("Mauvais token, veuillez réessayer.");
  47. }
  48. }
  49. if(!empty($_GET['deco'])) {
  50. session_destroy();
  51. }
  52. if(is_file('data/config')) {
  53. $config = unserialize(gzinflate(base64_decode(file_get_contents('data/config'))));
  54. if(!empty($_GET['code']) && $_GET['code'] == $config[0]) {
  55. $_SESSION['admin'] = true;
  56. }
  57. }
  58. $search = false;
  59. if(!empty($_POST['start_search']) && !empty($_POST['end_search']) & !empty($_POST['token'])) {
  60. if(check_token(600, 'defivelib')) {
  61. $search = true;
  62. $data = array_filter($data, "search");
  63. }
  64. else {
  65. exit("Mauvais token, veuillez réessayer.");
  66. }
  67. }
  68. if((!empty($_POST['time_min']) || !empty($_POST['time_sec'])) && !empty($_POST['start']) && !empty($_POST['end'])) {
  69. if(check_token(600, 'defivelib')) {
  70. $min = (!empty($_POST['time_min'])) ? (int) $_POST['time_min'] : 0;
  71. $sec = (!empty($_POST['time_sec'])) ? (int) $_POST['time_sec'] : 0;
  72. $pseudo = (!empty($_POST['pseudo'])) ? $_POST['pseudo'] : "Anonyme";
  73. $data[] = array("date"=>time(), "start"=>(int) $_POST['start'], "end"=>(int) $_POST['end'], "min"=>$min, "sec"=>$sec, "pseudo"=>$pseudo);
  74. // TODO : Upload + taille max de l'upload
  75. if(count($data) == 1 || $min != $data[count($data)-2]['min'] || $sec != $data[count($data)-2]['sec'] || $_POST['start'] != $data[count($data)-2]['start'] || $_POST['end'] != $data[count($data)-2]['end'] || $pseudo != $data[count($data)-2]['pseudo']) {
  76. file_put_contents('data/data', base64_encode(gzdeflate(serialize($data))));
  77. }
  78. }
  79. else {
  80. exit("Mauvais token, veuillez réessayer.");
  81. }
  82. }
  83. $token = generate_token('defivelib');
  84. ?>
  85. <!DOCTYPE html>
  86. <html lang="fr">
  87. <head>
  88. <meta charset="utf-8">
  89. <title>DéfiVélib</title>
  90. <meta name="author" content="phyks">
  91. <style type="text/css" media="screen">
  92. html, body {
  93. margin: 0;
  94. padding: 0;
  95. width: 100%;
  96. height: 100%;
  97. font-family: sans-serif;
  98. background-color: lightgrey;
  99. background-image: url();
  100. }
  101. #main {
  102. width: 50%;
  103. min-height: 50%;
  104. margin: auto;
  105. background-color: white;
  106. text-align: center;
  107. padding: 1em;
  108. -moz-box-sizing: border-box; -webkit-box-sizing: border-box; box-sizing: border-box;
  109. }
  110. @media (max-width: 1050px) {
  111. #main {
  112. width: 100% !important;
  113. }
  114. }
  115. #main h1 {
  116. margin: 0;
  117. border: 1px solid black;
  118. background-color: rgb(117, 170, 39);
  119. }
  120. #main h2 {
  121. text-align: left;
  122. }
  123. #main h1 a {
  124. color: black;
  125. text-decoration: none;
  126. font-size: 1.5em;
  127. }
  128. table td, th {
  129. padding: 0.5em;
  130. border: 1px solid black;
  131. }
  132. table th {
  133. background-color: rgba(117, 170, 39, 0.4);
  134. }
  135. table {
  136. border-collapse: collapse;
  137. margin: auto;
  138. }
  139. fieldset {
  140. background-color: rgba(17, 78, 121, 0.1);
  141. }
  142. #disclaimer {
  143. font-size: 0.75em;
  144. }
  145. #disclaimer p {
  146. text-align: left;
  147. }
  148. </style>
  149. </head>
  150. <body>
  151. <div id="main">
  152. <h1><a href="index.php">DéfiVélib</a></h1>
  153. <?php
  154. if(!is_dir('data/')) {
  155. mkdir('data/');
  156. }
  157. if(!is_file('data/config')) //First run
  158. {
  159. //Define a new synchronisation code
  160. $code_synchro = substr(sha1(rand(0,30).time().rand(0,30)),0,10);
  161. file_put_contents('data/config', base64_encode(gzdeflate(serialize(array($code_synchro))))); //Save it in data/data file
  162. $_GET['code'] = $code_synchro;
  163. echo "<p>
  164. Définition du code de synchronisation.<br/>
  165. Vous pouvez désormais mettre à jour la liste des stations en visitant l'adresse suivante (update URL) :<br/>
  166. <a href='http://" . $_SERVER["SERVER_NAME"].$_SERVER['REQUEST_URI']."?update=1&code=".$code_synchro."'>http://" . $_SERVER["SERVER_NAME"].$_SERVER['REQUEST_URI']."?update=1&code=".$code_synchro."</a>
  167. </p>
  168. <p>
  169. Il est possible d'automatiser la tâche via une tâche cron. Par exemple (see README) :<br/>
  170. * * * * * wget -q -O <a href='http://" . $_SERVER["SERVER_NAME"].$_SERVER['REQUEST_URI']."?update=1&code=".$code_synchro."'>http://" . $_SERVER["SERVER_NAME"].$_SERVER['REQUEST_URI']."?update=1&code=".$code_synchro."</a> #Commande de mise a jour des stations de velib
  171. </p>";
  172. }
  173. if(!empty($_GET['update']) || !empty($code_synchro)) //If we want to make an update (or first run)
  174. {
  175. if(empty($code_synchro) && is_file('data/config')) //If not first run, get the synchronisation code from data file
  176. {
  177. $data = unserialize(gzinflate(base64_decode(file_get_contents('data/config'))));
  178. $code_synchro = $data[0];
  179. }
  180. if(!empty($_GET['code']) && $_GET['code'] == $code_synchro) //Once we have the code and it is correct
  181. {
  182. $stations_xml = simplexml_load_file('http://www.velib.paris.fr/service/carto');
  183. $liste_stations = array();
  184. foreach($stations_xml->markers->marker as $station) {
  185. $liste_stations[(int) $station['number']] = array('name'=>(string) $station['name'], 'address'=>(string) $station['fullAddress'], 'lat'=>(float) $station['lat'], 'lng'=>(float) $station['lng']);
  186. }
  187. file_put_contents('data/stations', base64_encode(gzdeflate(serialize($liste_stations))));
  188. echo "<p>Mise à jour de la liste des stations effectuée avec succès (Update successful).</p>";
  189. }
  190. else
  191. {
  192. echo "<p>Mauvais code de vérification (Error : bad synchronisation code). Veuillez réessayer la mise à jour. Se référer au README pour plus d'informations sur la mise à jour.</p>";
  193. }
  194. echo "<p><a href='index.php'>Retourner à l'application (Back to index)</a></p></body></html>";
  195. exit();
  196. }
  197. $liste_stations = unserialize(gzinflate(base64_decode(file_get_contents('data/stations'))));
  198. ?>
  199. <div id="disclaimer">
  200. <h2>Disclaimer</h2>
  201. <p>Les temps rentrés sur cette page ne sont qu'indicatifs et sont fournis par les internautes eux-mêmes. Ils peuvent donc ne pas refléter les temps réels de parcours. En particulier, il est important de rappeler que le code de la route s'applique aussi aux vélos et que l'obtention d'un meilleur temps ne doit pas se faire au détriment du respect du code de la route.</p>
  202. <p>Le respect des données personnelles étant particulièrement important, ce script ne conserve aucune information particulière si vous ne souhaitez pas en renseigner. Votre adresse IP est néanmoins stockée dans les logs de connexion au serveur, comme pour tout serveur web, conformément à l'article 6 de la LCEN.</p>
  203. <p><a href="README.md">Plus d'informations sur DefiVelib</a></p>
  204. </div>
  205. <h2>Ajouter un trajet</h2>
  206. <form method="post" action="index.php"> <!-- enctype="multipart/form-data"-->
  207. <fieldset>
  208. <legend>Trajet</legend>
  209. <p><label name="start">Station de départ : </label>
  210. <select name="start" id="start">
  211. <?php
  212. foreach($liste_stations as $key=>$station) {
  213. if(!empty($_POST['start_search']) && $_POST['start_search'] == $key)
  214. $selected = "selected";
  215. else
  216. $selected = "";
  217. echo "<option value=\"".$key."\" ".$selected.">".$station['name']."</option>";
  218. }
  219. ?>
  220. </select>
  221. </p>
  222. <p><label for="end">Station d'arrivée : </label>
  223. <select name="end" id="end">
  224. <?php
  225. foreach($liste_stations as $key=>$station) {
  226. if(!empty($_POST['end_search']) && $_POST['end_search'] == $key)
  227. $selected = "selected";
  228. else
  229. $selected = "";
  230. echo "<option value=\"".$key."\" ".$selected.">".$station['name']."</option>";
  231. }
  232. ?>
  233. </select>
  234. </p>
  235. <p><label for="time_min">Durée du trajet : </label><input type="int" name="time_min" id="time_min" size="2"/>min <input type="int" name="time_sec" id="time_sec" size="2"/>s</p>
  236. </fieldset>
  237. <fieldset>
  238. <legend>Informations</legend>
  239. <p><label for="pseudo">Votre pseudo (optionnel) : </label><input type="text" name="pseudo" id="pseudo"/></p>
  240. <!-- <p><label for="photo">Photo du ticket (? max) : </label><input type="file" name="photo" id="photo"></p>-->
  241. </fieldset>
  242. <p>
  243. <input type="submit" value="Envoyer"/>
  244. <input type="hidden" name="token" value="<?php echo $token; ?>"/>
  245. <!-- <input type="hidden" name="MAX_FILE_SIZE" value="2097152">-->
  246. </p>
  247. </form>
  248. <h2><?php if($search) {?>Résultats<?php } else {?>Derniers trajets ajoutés<?php }?></h2>
  249. <?php
  250. if(!empty($data)) {
  251. ?>
  252. <table>
  253. <tr>
  254. <th>Date</th>
  255. <th>Départ</th>
  256. <th>Arrivée</th>
  257. <th>Temps</th>
  258. <th>Pseudo</th>
  259. <?php
  260. if(!empty($_SESSION['admin'])) {
  261. ?>
  262. <th>Supprimer</th>
  263. <?php
  264. }
  265. ?>
  266. </tr>
  267. <?php
  268. if($search) {
  269. $min = array();
  270. $sec = array();
  271. foreach($data as $key => $result) {
  272. $min[$key] = $result['min'];
  273. $sec[$key] = $result['sec'];
  274. }
  275. array_multisort($min, SORT_DESC, $sec, SORT_DESC, $data);
  276. foreach($data as $key=>$result) {
  277. if(!empty($_SESSION['admin'])) {
  278. $delete = "<td><a href=\"?suppr=".$key."&token=".$token."\">Supprimer</a></td>";
  279. }
  280. else {
  281. $delete = "";
  282. }
  283. echo "<tr><td>".date('d/m/Y à H:i', $result['date'])."</td><td>".htmlspecialchars($liste_stations[$result['start']]['name'])."</td><td>".htmlspecialchars($liste_stations[$result['end']]['name'])."</td><td>".(int) $result['min']."min ".(int) $result['sec']."s</td><td>".htmlspecialchars($result['pseudo'])."</td>".$delete."</tr>";
  284. }
  285. }
  286. else {
  287. for($i = count($data) - 1; $i >= max(count($data) - 10, 0); $i--) {
  288. if(!empty($_SESSION['admin'])) {
  289. $delete = "<td><a href=\"?suppr=".$i."&token=".$token."\">Supprimer</a></td>";
  290. }
  291. else {
  292. $delete = "";
  293. }
  294. echo "<tr><td>".date('d/m/Y à H:i', $data[$i]['date'])."</td><td>".htmlspecialchars($liste_stations[$data[$i]['start']]['name'])."</td><td>".htmlspecialchars($liste_stations[$data[$i]['end']]['name'])."</td><td>".(int) $data[$i]['min']."min ".(int) $data[$i]['sec']."s</td><td>".htmlspecialchars($data[$i]['pseudo'])."</td>".$delete."</tr>";
  295. }
  296. }
  297. ?>
  298. </table>
  299. <?php
  300. }
  301. else {
  302. ?>
  303. <p>Aucun trajet enregistré.</p>
  304. <?php
  305. }
  306. ?>
  307. <h2>Recherche de trajets</h2>
  308. <form method="post" action="index.php">
  309. <fieldset>
  310. <p><label name="start_search">Station de départ : </label>
  311. <select name="start_search" id="start_search">
  312. <?php
  313. foreach($liste_stations as $key=>$station) {
  314. if(!empty($_POST['start_search']) && $_POST['start_search'] == $key)
  315. $selected = "selected";
  316. else
  317. $selected = "";
  318. echo "<option value=\"".$key."\" ".$selected.">".$station['name']."</option>";
  319. }
  320. ?>
  321. </select>
  322. </p>
  323. <p><label for="end_search">Station d'arrivée : </label>
  324. <select name="end_search" id="end_search">
  325. <?php
  326. foreach($liste_stations as $key=>$station) {
  327. if(!empty($_POST['end_search']) && $_POST['end_search'] == $key)
  328. $selected = "selected";
  329. else
  330. $selected = "";
  331. echo "<option value=\"".$key."\" ".$selected.">".$station['name']."</option>";
  332. }
  333. ?>
  334. </select>
  335. </p>
  336. </fieldset>
  337. <p>
  338. <input type="submit" value="Rechercher"/>
  339. <input type="hidden" name="token" value="<?php echo $token; ?>"/>
  340. </p>
  341. </form>
  342. <?php
  343. if(!empty($_SESSION['admin'])) {
  344. ?>
  345. <p><a href="?deco=1">Déconnexion</a></p>
  346. <?php
  347. }
  348. ?>
  349. </div>
  350. </body>
  351. </html>