RideLogic.pm 13 KB

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