RideLogic.pm 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376
  1. package RideLogic;
  2. use strict;
  3. use POSIX;
  4. require Exporter;
  5. use vars qw($VERSION @ISA @EXPORT @EXPORT_OK %EXPORT_TAGS);
  6. $VERSION = "0.01";
  7. @ISA = qw(Exporter);
  8. #@EXPORT = qw( daemonize
  9. # read_config
  10. # strip_whitespace
  11. # try
  12. # catch );
  13. #@EXPORT_OK = qw( daemonize
  14. # read_config
  15. # strip_whitespace
  16. # try
  17. # catch );
  18. #%EXPORT_TAGS = ( DEFAULT => [qw( &daemonize
  19. # &read_config
  20. # &strip_whitespace
  21. # &try
  22. # &catch )] );
  23. @EXPORT = qw( audit_user_card_start
  24. audit_user_card_end
  25. audit_user_pass_start
  26. audit_user_pass_end
  27. audit_users_start
  28. audit_users_end
  29. audit_admins_start
  30. audit_admins_end
  31. daemonize
  32. read_config
  33. strip_whitespace
  34. try
  35. catch );
  36. @EXPORT_OK = qw( audit_user_card_start
  37. audit_user_card_end
  38. audit_user_pass_start
  39. audit_user_pass_end
  40. audit_users_start
  41. audit_users_end
  42. audit_admins_start
  43. audit_admins_end
  44. daemonize
  45. read_config
  46. strip_whitespace
  47. try
  48. catch );
  49. %EXPORT_TAGS = ( DEFAULT => [qw( &audit_user_card_start
  50. &audit_user_card_end
  51. &audit_user_pass_start
  52. &audit_user_pass_end
  53. &audit_users_start
  54. &audit_users_end
  55. &audit_admins_start
  56. &audit_admins_end
  57. &daemonize
  58. &read_config
  59. &strip_whitespace
  60. &try
  61. &catch )] );
  62. our @ISA = qw(Exporter);
  63. our $VERSION="0.1";
  64. our $RIDELOGIC_DAEMON_CONF="/etc/ridelogic/daemon.conf";
  65. our $RIDELOGIC_DAEMON_LOG_DIR="/var/log/ridelogic";
  66. our $RIDELOGIC_DAEMON_PID_DIR="/var/run/ridelogic";
  67. our $RIDELOGIC_API_CONF="/etc/ridelogic/api.conf";
  68. sub audit_user_pass_start {
  69. my $dbh = shift;
  70. my $pass_id = shift;
  71. my $comment = shift;
  72. my @field = qw( user_pass_id logical_card_id issued activated firstused lastused nrides_orig nrides_remain nday_orig nday_expiration active rule queue_order comment expired paytype deactivated );
  73. my $ins_query = " insert into audit_user_pass ( timestamp, comment, old_" . $field[0];
  74. my $tail_ins_query = " select now(), ?, " . ($pass_id ? $field[0] : "null");
  75. for (my $i=1; $i<scalar(@field); $i++) {
  76. $ins_query .= ", old_" . $field[$i];
  77. $tail_ins_query .= ", " . ($pass_id ? $field[$i] : "null" );
  78. }
  79. $ins_query .= ") ";
  80. $tail_ins_query .= ($pass_id ? " from user_pass where user_pass_id = ?" : "");
  81. $ins_query .= $tail_ins_query;
  82. my $query = $dbh->prepare($ins_query);
  83. my $response = ($pass_id ? $query->execute($comment, $pass_id) : $query->execute($comment));
  84. return ($dbh->last_insert_id(undef, undef, undef, undef));
  85. }
  86. sub audit_user_pass_end {
  87. my $dbh = shift;
  88. my $pass_id = shift;
  89. my $audit_id = shift;
  90. my @field = qw( user_pass_id logical_card_id issued activated firstused lastused nrides_orig nrides_remain nday_orig nday_expiration active rule queue_order comment expired paytype deactivated );
  91. my $up_query = " update audit_user_pass, user_pass set audit_user_pass.new_" . $field[0] . " = user_pass." . $field[0];
  92. for (my $i=1; $i<scalar(@field); $i++) {
  93. $up_query .= ", audit_user_pass.new_".$field[$i] . " = user_pass." . $field[$i];
  94. }
  95. $up_query .= " where audit_user_pass.audit_user_pass_id = ? and user_pass.user_pass_id = ? ";
  96. my $query = $dbh->prepare($up_query);
  97. my $response = $query->execute($audit_id, $pass_id);
  98. }
  99. sub audit_user_card_start {
  100. my $dbh = shift;
  101. my $card_id = shift;
  102. my $comment = shift;
  103. my @field = qw( logical_card_id mag_token rfid_token comment lastused userid issued firstused group_id active deactivated issuetype );
  104. my $ins_query = " insert into audit_user_card ( timestamp, comment, old_" . $field[0];
  105. my $tail_ins_query = " select now(), ?, " . ($card_id ? $field[0] : "null");
  106. for (my $i=1; $i<scalar(@field); $i++) {
  107. $ins_query .= ", old_" . $field[$i];
  108. $tail_ins_query .= ", " . ($card_id ? $field[$i] : "null");
  109. }
  110. $ins_query .= ") ";
  111. $tail_ins_query .= ($card_id ? " from user_card where logical_card_id = ?" : "");
  112. $ins_query .= $tail_ins_query;
  113. my $query = $dbh->prepare($ins_query);
  114. my $response = ($card_id ? $query->execute($comment, $card_id) : $query->execute($comment) );
  115. return ($dbh->last_insert_id(undef, undef, undef, undef));
  116. }
  117. sub audit_user_card_end {
  118. my $dbh = shift;
  119. my $card_id = shift;
  120. my $audit_id = shift;
  121. my @field = qw( logical_card_id mag_token rfid_token comment lastused userid issued firstused group_id active deactivated issuetype );
  122. my $up_query = " update audit_user_card, user_card set audit_user_card.new_" . $field[0] . " = user_card." . $field[0];
  123. for (my $i=1; $i<scalar(@field); $i++) {
  124. $up_query .= ", audit_user_card.new_".$field[$i] . " = user_card." . $field[$i];
  125. }
  126. $up_query .= " where audit_user_card.audit_user_card_id = ? and user_card.logical_card_id = ? ";
  127. my $query = $dbh->prepare($up_query);
  128. my $response = $query->execute($audit_id, $card_id);
  129. }
  130. sub audit_users_start {
  131. my $dbh = shift;
  132. my $userid = shift;
  133. my $comment = shift;
  134. my @field = qw( username passwordhash userid comment first_name last_name phone email address city state zip created active
  135. shipping_address shipping_city shipping_state shipping_zip shipping_name shipping_country_code shipping_country_name reset_attempts );
  136. my $ins_query = " insert into audit_users ( timestamp, comment, old_" . $field[0];
  137. my $tail_ins_query = " select now(), ?, " . ($userid ? $field[0] : "null");
  138. for (my $i=1; $i<scalar(@field); $i++) {
  139. $ins_query .= ", old_".$field[$i];
  140. $tail_ins_query .= ", " . ($userid ? $field[$i] : "null");
  141. }
  142. $ins_query .= ") ";
  143. $tail_ins_query .= ($userid ? " from users where userid = ?" : "");
  144. $ins_query .= $tail_ins_query;
  145. my $query = $dbh->prepare($ins_query);
  146. my $response = ($userid ? $query->execute($comment, $userid) : $query->execute($comment) );
  147. return ($dbh->last_insert_id(undef, undef, undef, undef));
  148. }
  149. sub audit_users_end {
  150. my $dbh = shift;
  151. my $userid = shift;
  152. my $audit_id = shift;
  153. my @field = qw( username passwordhash userid comment first_name last_name phone email address city state zip created active
  154. shipping_address shipping_city shipping_state shipping_zip shipping_name shipping_country_code shipping_country_name reset_attempts );
  155. my $up_query = " update audit_users, users set audit_users.new_" . $field[0] . " = users." . $field[0];
  156. for (my $i=1; $i<scalar(@field); $i++) {
  157. $up_query .= ", audit_users.new_".$field[$i] . " = users." . $field[$i];
  158. }
  159. $up_query .= " where audit_users.audit_users_id = ? and users.userid = ? ";
  160. my $query = $dbh->prepare($up_query);
  161. my $response = $query->execute($audit_id, $userid);
  162. }
  163. sub audit_admins_start {
  164. my $dbh = shift;
  165. my $userid = shift;
  166. my $comment = shift;
  167. my @field = qw( username password userid );
  168. my $ins_query = " insert into audit_admins ( timestamp, comment, old_" . $field[0];
  169. my $tail_ins_query = " select now(), ?, " . ($userid ? $field[0] : "null");
  170. for (my $i=1; $i<scalar(@field); $i++) {
  171. $ins_query .= ", old_".$field[$i];
  172. $tail_ins_query .= ", " . ($userid ? $field[$i] : "null");
  173. }
  174. $ins_query .= ") ";
  175. $tail_ins_query .= ($userid ? " from admins where userid = ?" : "");
  176. $ins_query .= $tail_ins_query;
  177. my $query = $dbh->prepare($ins_query);
  178. my $response = ($userid ? $query->execute($comment, $userid) : $query->execute($comment) );
  179. return ($dbh->last_insert_id(undef, undef, undef, undef));
  180. }
  181. sub audit_admins_end {
  182. my $dbh = shift;
  183. my $userid = shift;
  184. my $audit_id = shift;
  185. my @field = qw( username password userid );
  186. my $up_query = " update audit_admins, admins set audit_admins.new_" . $field[0] . " = admins." . $field[0];
  187. for (my $i=1; $i<scalar(@field); $i++) {
  188. $up_query .= ", audit_admins.new_".$field[$i] . " = admins." . $field[$i];
  189. }
  190. $up_query .= " where audit_admins.audit_admins_id = ? and admins.userid = ? ";
  191. my $query = $dbh->prepare($up_query);
  192. my $response = $query->execute($audit_id, $userid);
  193. }
  194. #----------------------------------------------Ugly exception handling logic using closures and anonymous functions----
  195. #-------------------------------------------This is in there to deal with the fact that CreditCall uses the die("error")
  196. #-------------------------------------------function instead of returning an error message in many cases...
  197. # This utility function returns the passed string sans any leading or trailing whitespace.
  198. #
  199. sub strip_whitespace
  200. {
  201. my $str = shift; #grab our first parameter
  202. $str =~ s/^\s+//; #strip leading whitespace
  203. $str =~ s/\s+$//; #strip trailing whitespace
  204. return $str; #return the improved string
  205. }
  206. # This function takes two coderef parameters, the second of which is usually an explicit call to the
  207. # 'catch' function which itself takes a coderef parameter. This allows the code employing this suite of
  208. # functions to look somewhat like a conventional exception handling mechanism:
  209. #
  210. # try
  211. # {
  212. # do_something_that_might_die();
  213. # }
  214. # catch
  215. # {
  216. # my $errmsg = $_;
  217. # log_the_error_message($errmsg);
  218. # perform_some_cleanup();
  219. # };
  220. #
  221. # DO NOT FORGET THAT LAST SEMICOLON, EVERYTHING GOES TO HELL IF YOU DO!
  222. #
  223. sub try(&$)
  224. {
  225. my ($attempt, $handler) = @_;
  226. eval
  227. {
  228. &$attempt;
  229. };
  230. if($@)
  231. {
  232. do_catch($handler);
  233. }
  234. }
  235. # This function strips off the whitespace from the exception message reported by die()
  236. # and places the result into the default variable such that the code in the catch block can
  237. # just examine $_ to figure out what the cause of the error is, or to display or log
  238. # the error message.
  239. #
  240. sub do_catch(&$)
  241. {
  242. my ($handler) = @_;
  243. local $_ = strip_whitespace($@);
  244. &$handler;
  245. }
  246. # This just takes an explicit coderef and returns it unharmed. The only
  247. # purpose of this is so the try/catch structure looks pretty and familiar.
  248. #
  249. sub catch(&) {$_[0]}
  250. sub read_config
  251. {
  252. my $cfg_file = shift;
  253. my $cfg_href = shift;
  254. try
  255. {
  256. open my $fh, "$cfg_file";
  257. while (<$fh>) {
  258. next if /^#/;
  259. chomp;
  260. s/^\s+//;
  261. s/\s+$//;
  262. next if !$_;
  263. my ($var, $val) = split(/=/, $_);
  264. $cfg_href->{$var} = $val;
  265. }
  266. close $fh;
  267. }
  268. catch
  269. {
  270. my $errmsg = $_;
  271. die "Error processing config file $cfg_file, '$errmsg'. exiting\n";
  272. };
  273. }
  274. sub daemonize
  275. {
  276. my $logfile = shift;
  277. my $pidfile = shift;
  278. my ($untainted_lf, $untainted_pf);
  279. $untainted_lf = $1 if ($logfile =~ /^([^\0]+)$/);
  280. $untainted_pf = $1 if ($pidfile =~ /^([^\0]+)$/);
  281. try
  282. {
  283. chdir '/';
  284. umask 0;
  285. open STDIN, '/dev/null';
  286. open STDOUT, '/dev/null';
  287. open STDERR, '/dev/null';
  288. my $pid = fork;
  289. exit if $pid;
  290. setsid;
  291. if ($untainted_lf)
  292. {
  293. close STDOUT;
  294. close STDERR;
  295. sysopen( STDOUT, $untainted_lf, O_WRONLY|O_APPEND|O_CREAT, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH );
  296. sysopen( STDERR, $untainted_lf, O_WRONLY|O_APPEND|O_CREAT, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH );
  297. }
  298. if ($untainted_pf)
  299. {
  300. sysopen( my $fh, $untainted_pf, O_WRONLY|O_APPEND|O_CREAT, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH );
  301. #open my $fh, ">$untainted_pf";
  302. print $fh $$, "\n";
  303. close $fh ;
  304. }
  305. }
  306. catch
  307. {
  308. my $errmsg = $_;
  309. die "Failed to daemonize: '$errmsg', exiting\n";
  310. };
  311. }
  312. return 1;