navigo.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521
  1. #include "navigo.h"
  2. #include "navigo_lists.h"
  3. #include "../../../metroflip_i.h"
  4. const char* get_navigo_transport_type(int type) {
  5. switch(type) {
  6. case BUS_URBAIN:
  7. return "Bus Urbain";
  8. case BUS_INTERURBAIN:
  9. return "Bus Interurbain";
  10. case METRO:
  11. return "Metro";
  12. case TRAM:
  13. return "Tram";
  14. case TRAIN:
  15. return "Train";
  16. case PARKING:
  17. return "Parking";
  18. default:
  19. return "Unknown";
  20. }
  21. }
  22. const char* get_navigo_service_provider(int provider) {
  23. switch(provider) {
  24. case 2:
  25. return "SNCF";
  26. case 3:
  27. return "RATP";
  28. case 4:
  29. return "IDF Mobilites";
  30. case 10:
  31. return "IDF Mobilites";
  32. case 115:
  33. return "CSO (VEOLIA)";
  34. case 116:
  35. return "R'Bus (VEOLIA)";
  36. case 156:
  37. return "Phebus";
  38. case 175:
  39. return "RATP (Veolia Transport Nanterre)";
  40. default:
  41. return "Unknown";
  42. }
  43. }
  44. const char* get_transition_type(int transition) {
  45. switch(transition) {
  46. case 1:
  47. return "Entry";
  48. case 2:
  49. return "Exit";
  50. case 4:
  51. return "Controle volant (a bord)";
  52. case 5:
  53. return "Test validation";
  54. case 6:
  55. return "Interchange - Entry";
  56. case 7:
  57. return "Interchange - Exit";
  58. case 9:
  59. return "Validation cancelled";
  60. case 10:
  61. return "Entry";
  62. case 11:
  63. return "Exit";
  64. case 13:
  65. return "Distribution";
  66. case 15:
  67. return "Invalidation";
  68. default: {
  69. char* transition_str = malloc(6 * sizeof(char));
  70. snprintf(transition_str, 6, "%d", transition);
  71. return transition_str;
  72. }
  73. }
  74. }
  75. const char* get_navigo_type(int type) {
  76. switch(type) {
  77. case NAVIGO_EASY:
  78. return "Navigo Easy";
  79. case NAVIGO_DECOUVERTE:
  80. return "Navigo Decouverte";
  81. case NAVIGO_STANDARD:
  82. return "Navigo Standard";
  83. case NAVIGO_INTEGRAL:
  84. return "Navigo Integral";
  85. case IMAGINE_R:
  86. return "Imagine R";
  87. default:
  88. return "Navigo";
  89. }
  90. }
  91. const char* get_navigo_tariff(int tariff) {
  92. switch(tariff) {
  93. case 0x0000:
  94. return "Navigo Mois";
  95. case 0x0001:
  96. return "Navigo Semaine";
  97. case 0x0002:
  98. return "Navigo Annuel";
  99. case 0x0003:
  100. return "Navigo Jour";
  101. case 0x0004:
  102. return "Imagine R Junior";
  103. case 0x0005:
  104. return "Imagine R Etudiant";
  105. case 0x000D:
  106. return "Navigo Jeunes Week-end";
  107. case 0x0015:
  108. return "Paris-Visite"; // Theoric
  109. case 0x1000:
  110. return "Navigo Liberte+";
  111. case 0x4000:
  112. return "Navigo Mois 75%%";
  113. case 0x4001:
  114. return "Navigo Semaine 75%%";
  115. case 0x4015:
  116. return "Paris-Visite (Enfant)"; // Theoric
  117. case 0x5000:
  118. return "Tickets T+";
  119. case 0x5004:
  120. return "Tickets OrlyBus"; // Theoric
  121. case 0x5005:
  122. return "Tickets RoissyBus"; // Theoric
  123. case 0x5006:
  124. return "Bus-Tram"; // Theoric
  125. case 0x5008:
  126. return "Metro-Train-RER"; // Theoric
  127. case 0x500b:
  128. return "Paris <> Aeroports"; // Theoric
  129. case 0x5010:
  130. return "Tickets T+ (Reduit)"; // Theoric
  131. case 0x5016:
  132. return "Bus-Tram (Reduit)"; // Theoric
  133. case 0x5018:
  134. return "Metro-Train-RER (Reduit)"; // Theoric
  135. case 0x501b:
  136. return "Paris <> Aeroports (Reduit)"; // Theoric
  137. case 0x8003:
  138. return "Navigo Solidarite Gratuit";
  139. default: {
  140. char* tariff_str = malloc(6 * sizeof(char));
  141. snprintf(tariff_str, 6, "%d", tariff);
  142. return tariff_str;
  143. }
  144. }
  145. }
  146. bool is_ticket_count_available(int tariff) {
  147. return tariff >= 0x5000 && tariff <= 0x501b;
  148. }
  149. const char* get_pay_method(int pay_method) {
  150. switch(pay_method) {
  151. case 0x30:
  152. return "Apple Pay";
  153. case 0x80:
  154. return "Debit PME";
  155. case 0x90:
  156. return "Cash";
  157. case 0xA0:
  158. return "Mobility Check";
  159. case 0xB3:
  160. return "Payment Card";
  161. case 0xA4:
  162. return "Check";
  163. case 0xA5:
  164. return "Vacation Check";
  165. case 0xB7:
  166. return "Telepayment";
  167. case 0xD0:
  168. return "Remote Payment";
  169. case 0xD7:
  170. return "Voucher, Prepayment, Exchange Voucher, Travel Voucher";
  171. case 0xD9:
  172. return "Discount Voucher";
  173. default:
  174. return "Unknown";
  175. }
  176. }
  177. const char* get_zones(int* zones) {
  178. if(zones[0] && zones[4]) {
  179. return "All Zones (1-5)";
  180. } else if(zones[0] && zones[3]) {
  181. return "Zones 1-4";
  182. } else if(zones[0] && zones[2]) {
  183. return "Zones 1-3";
  184. } else if(zones[0] && zones[1]) {
  185. return "Zones 1-2";
  186. } else if(zones[0]) {
  187. return "Zone 1";
  188. } else if(zones[1] && zones[4]) {
  189. return "Zones 2-5";
  190. } else if(zones[1] && zones[3]) {
  191. return "Zones 2-4";
  192. } else if(zones[1] && zones[2]) {
  193. return "Zones 2-3";
  194. } else if(zones[1]) {
  195. return "Zone 2";
  196. } else if(zones[2] && zones[4]) {
  197. return "Zones 3-5";
  198. } else if(zones[2] && zones[3]) {
  199. return "Zones 3-4";
  200. } else if(zones[2]) {
  201. return "Zone 3";
  202. } else if(zones[3] && zones[4]) {
  203. return "Zones 4-5";
  204. } else if(zones[3]) {
  205. return "Zone 4";
  206. } else if(zones[4]) {
  207. return "Zone 5";
  208. } else {
  209. return "Unknown";
  210. }
  211. }
  212. const char* get_intercode_version(int version) {
  213. // version is a 6 bits int
  214. // if the first 3 bits are 000, it's a 1.x version
  215. // if the first 3 bits are 001, it's a 2.x version
  216. // else, it's unknown
  217. int major = (version >> 3) & 0x07;
  218. if(major == 0) {
  219. return "Intercode I";
  220. } else if(major == 1) {
  221. return "Intercode II";
  222. }
  223. return "Unknown";
  224. }
  225. int get_intercode_subversion(int version) {
  226. // subversion is a 3 bits int
  227. return version & 0x07;
  228. }
  229. const char* get_navigo_metro_station(int station_group_id, int station_id) {
  230. // Use NAVIGO_H constants
  231. if(station_group_id < 32 && station_id < 16) {
  232. return NAVIGO_METRO_STATION_LIST[station_group_id][station_id];
  233. }
  234. // cast station_group_id-station_id to a string
  235. char* station = malloc(12 * sizeof(char));
  236. if(!station) {
  237. return "Unknown";
  238. }
  239. snprintf(station, 10, "%d-%d", station_group_id, station_id);
  240. return station;
  241. }
  242. const char* get_navigo_train_line(int station_group_id) {
  243. if(station_group_id < 77) {
  244. return NAVIGO_TRAIN_LINES_LIST[station_group_id];
  245. }
  246. return "Unknown";
  247. }
  248. const char* get_navigo_train_station(int station_group_id, int station_id) {
  249. if(station_group_id < 77 && station_id < 19) {
  250. return NAVIGO_TRAIN_STATION_LIST[station_group_id][station_id];
  251. }
  252. // cast station_group_id-station_id to a string
  253. char* station = malloc(12 * sizeof(char));
  254. if(!station) {
  255. return "Unknown";
  256. }
  257. snprintf(station, 10, "%d-%d", station_group_id, station_id);
  258. return station;
  259. }
  260. const char* get_navigo_tram_line(int route_number) {
  261. switch(route_number) {
  262. case 16:
  263. return "T6";
  264. default: {
  265. char* line = malloc(3 * sizeof(char));
  266. if(!line) {
  267. return "Unknown";
  268. }
  269. snprintf(line, 3, "T%d", route_number);
  270. return line;
  271. }
  272. }
  273. }
  274. void show_navigo_event_info(
  275. NavigoCardEvent* event,
  276. NavigoCardContract* contracts,
  277. FuriString* parsed_data) {
  278. if(event->used_contract == 0) {
  279. furi_string_cat_printf(parsed_data, "No event data\n");
  280. return;
  281. }
  282. if(event->transport_type == BUS_URBAIN || event->transport_type == BUS_INTERURBAIN ||
  283. event->transport_type == METRO || event->transport_type == TRAM) {
  284. if(event->route_number_available) {
  285. if(event->transport_type == METRO && event->route_number == 103) {
  286. furi_string_cat_printf(
  287. parsed_data,
  288. "%s 3 bis\n%s\n",
  289. get_navigo_transport_type(event->transport_type),
  290. get_transition_type(event->transition));
  291. } else if(event->transport_type == TRAM) {
  292. furi_string_cat_printf(
  293. parsed_data,
  294. "%s %s\n%s\n",
  295. get_navigo_transport_type(event->transport_type),
  296. get_navigo_tram_line(event->route_number),
  297. get_transition_type(event->transition));
  298. } else {
  299. furi_string_cat_printf(
  300. parsed_data,
  301. "%s %d\n%s\n",
  302. get_navigo_transport_type(event->transport_type),
  303. event->route_number,
  304. get_transition_type(event->transition));
  305. }
  306. } else {
  307. furi_string_cat_printf(
  308. parsed_data,
  309. "%s\n%s\n",
  310. get_navigo_transport_type(event->transport_type),
  311. get_transition_type(event->transition));
  312. }
  313. furi_string_cat_printf(
  314. parsed_data,
  315. "Transporter: %s\n",
  316. get_navigo_service_provider(event->service_provider));
  317. if(event->transport_type == METRO) {
  318. furi_string_cat_printf(
  319. parsed_data,
  320. "Station: %s\nSector: %s\n",
  321. get_navigo_metro_station(event->station_group_id, event->station_id),
  322. get_navigo_metro_station(event->station_group_id, 0));
  323. } else {
  324. furi_string_cat_printf(
  325. parsed_data, "Station ID: %d-%d\n", event->station_group_id, event->station_id);
  326. }
  327. if(event->location_gate_available) {
  328. furi_string_cat_printf(parsed_data, "Gate: %d\n", event->location_gate);
  329. }
  330. if(event->device_available) {
  331. if(event->transport_type == BUS_URBAIN || event->transport_type == BUS_INTERURBAIN) {
  332. const char* side = event->side == 0 ? "right" : "left";
  333. furi_string_cat_printf(parsed_data, "Door: %d\nSide: %s\n", event->door, side);
  334. } else {
  335. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  336. }
  337. }
  338. if(event->mission_available) {
  339. furi_string_cat_printf(parsed_data, "Mission: %d\n", event->mission);
  340. }
  341. if(event->vehicle_id_available) {
  342. furi_string_cat_printf(parsed_data, "Vehicle: %d\n", event->vehicle_id);
  343. }
  344. if(event->used_contract_available) {
  345. furi_string_cat_printf(
  346. parsed_data,
  347. "Contract: %d - %s\n",
  348. event->used_contract,
  349. get_navigo_tariff(contracts[event->used_contract - 1].tariff));
  350. }
  351. locale_format_datetime_cat(parsed_data, &event->date, true);
  352. furi_string_cat_printf(parsed_data, "\n");
  353. } else if(event->transport_type == TRAIN) {
  354. if(event->route_number_available) {
  355. furi_string_cat_printf(
  356. parsed_data,
  357. "RER %c\n%s\n",
  358. (65 + event->route_number - 17),
  359. get_transition_type(event->transition));
  360. } else {
  361. furi_string_cat_printf(
  362. parsed_data,
  363. "%s %s\n%s\n",
  364. get_navigo_transport_type(event->transport_type),
  365. get_navigo_train_line(event->station_group_id),
  366. get_transition_type(event->transition));
  367. }
  368. furi_string_cat_printf(
  369. parsed_data,
  370. "Transporter: %s\n",
  371. get_navigo_service_provider(event->service_provider));
  372. furi_string_cat_printf(
  373. parsed_data,
  374. "Station: %s\n",
  375. get_navigo_train_station(event->station_group_id, event->station_id));
  376. /* if(event->route_number_available) {
  377. furi_string_cat_printf(parsed_data, "Route: %d\n", event->route_number);
  378. } */
  379. if(event->location_gate_available) {
  380. furi_string_cat_printf(parsed_data, "Gate: %d\n", event->location_gate);
  381. }
  382. if(event->device_available) {
  383. if(event->service_provider == 2) {
  384. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device & 0xFF);
  385. } else {
  386. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  387. }
  388. }
  389. if(event->mission_available) {
  390. furi_string_cat_printf(parsed_data, "Mission: %d\n", event->mission);
  391. }
  392. if(event->vehicle_id_available) {
  393. furi_string_cat_printf(parsed_data, "Vehicle: %d\n", event->vehicle_id);
  394. }
  395. if(event->used_contract_available) {
  396. furi_string_cat_printf(
  397. parsed_data,
  398. "Contract: %d - %s\n",
  399. event->used_contract,
  400. get_navigo_tariff(contracts[event->used_contract - 1].tariff));
  401. }
  402. locale_format_datetime_cat(parsed_data, &event->date, true);
  403. furi_string_cat_printf(parsed_data, "\n");
  404. } else {
  405. furi_string_cat_printf(
  406. parsed_data,
  407. "%s - %s\n",
  408. get_navigo_transport_type(event->transport_type),
  409. get_transition_type(event->transition));
  410. furi_string_cat_printf(
  411. parsed_data,
  412. "Transporter: %s\n",
  413. get_navigo_service_provider(event->service_provider));
  414. furi_string_cat_printf(
  415. parsed_data, "Station ID: %d-%d\n", event->station_group_id, event->station_id);
  416. if(event->location_gate_available) {
  417. furi_string_cat_printf(parsed_data, "Gate: %d\n", event->location_gate);
  418. }
  419. if(event->device_available) {
  420. furi_string_cat_printf(parsed_data, "Device: %d\n", event->device);
  421. }
  422. if(event->mission_available) {
  423. furi_string_cat_printf(parsed_data, "Mission: %d\n", event->mission);
  424. }
  425. if(event->vehicle_id_available) {
  426. furi_string_cat_printf(parsed_data, "Vehicle: %d\n", event->vehicle_id);
  427. }
  428. if(event->used_contract_available) {
  429. furi_string_cat_printf(
  430. parsed_data,
  431. "Contract: %d - %s\n",
  432. event->used_contract,
  433. get_navigo_tariff(contracts[event->used_contract - 1].tariff));
  434. }
  435. locale_format_datetime_cat(parsed_data, &event->date, true);
  436. furi_string_cat_printf(parsed_data, "\n");
  437. }
  438. }
  439. void show_navigo_contract_info(NavigoCardContract* contract, FuriString* parsed_data) {
  440. // Core type and ticket info
  441. furi_string_cat_printf(parsed_data, "Type: %s\n", get_navigo_tariff(contract->tariff));
  442. if(is_ticket_count_available(contract->tariff)) {
  443. furi_string_cat_printf(parsed_data, "Remaining Tickets: %d\n", contract->counter.count);
  444. }
  445. // Validity period
  446. furi_string_cat_printf(parsed_data, "Valid from: ");
  447. locale_format_datetime_cat(parsed_data, &contract->start_date, false);
  448. furi_string_cat_printf(parsed_data, "\n");
  449. if(contract->end_date_available) {
  450. furi_string_cat_printf(parsed_data, "\nto: ");
  451. locale_format_datetime_cat(parsed_data, &contract->end_date, false);
  452. furi_string_cat_printf(parsed_data, "\n");
  453. }
  454. // Serial number (if available)
  455. if(contract->serial_number_available) {
  456. furi_string_cat_printf(parsed_data, "TCN Number: %d\n", contract->serial_number);
  457. }
  458. // Payment and pricing details
  459. if(contract->pay_method_available) {
  460. furi_string_cat_printf(
  461. parsed_data, "Payment Method: %s\n", get_pay_method(contract->pay_method));
  462. }
  463. if(contract->price_amount_available) {
  464. furi_string_cat_printf(parsed_data, "Amount: %.2f EUR\n", contract->price_amount);
  465. }
  466. // Zone and sales details
  467. if(contract->zones_available) {
  468. furi_string_cat_printf(parsed_data, "%s\n", get_zones(contract->zones));
  469. }
  470. furi_string_cat_printf(parsed_data, "Sold on: ");
  471. locale_format_datetime_cat(parsed_data, &contract->sale_date, false);
  472. furi_string_cat_printf(parsed_data, "\n");
  473. furi_string_cat_printf(
  474. parsed_data, "Sales Agent: %s\n", get_navigo_service_provider(contract->sale_agent));
  475. furi_string_cat_printf(parsed_data, "Sales Terminal: %d\n", contract->sale_device);
  476. // Status and authenticity
  477. if(contract->status == 1) {
  478. furi_string_cat_printf(parsed_data, "Status: OK\n");
  479. } else {
  480. furi_string_cat_printf(parsed_data, "Status: Unknown (%d)\n", contract->status);
  481. }
  482. furi_string_cat_printf(parsed_data, "Authenticity Code: %d\n", contract->authenticator);
  483. }
  484. void show_navigo_environment_info(NavigoCardEnv* environment, FuriString* parsed_data) {
  485. furi_string_cat_printf(
  486. parsed_data,
  487. "App Version: %s - v%d\n",
  488. get_intercode_version(environment->app_version),
  489. get_intercode_subversion(environment->app_version));
  490. furi_string_cat_printf(
  491. parsed_data, "Country: %s\n", get_country_string(environment->country_num));
  492. furi_string_cat_printf(
  493. parsed_data,
  494. "Network: %s\n",
  495. get_network_string(guess_card_type(environment->country_num, environment->network_num)));
  496. furi_string_cat_printf(parsed_data, "End of validity:\n");
  497. locale_format_datetime_cat(parsed_data, &environment->end_dt, false);
  498. furi_string_cat_printf(parsed_data, "\n");
  499. }