# This code was forked from the LiveJournal project owned and operated # by Live Journal, Inc. The code has been modified and expanded by # Dreamwidth Studios, LLC. These files were originally licensed under # the terms of the license supplied by Live Journal, Inc, which can # currently be found at: # # http://code.livejournal.org/trac/livejournal/browser/trunk/LICENSE-LiveJournal.txt # # In accordance with the original license, this code and all its # modifications are provided under the GNU General Public License. # A copy of that license can be found in the LICENSE file included as # part of this distribution. package LJ::Worker::TheSchwartz; use strict; use base "LJ::Worker", "Exporter"; require "$ENV{LJHOME}/cgi-bin/ljlib.pl"; use Getopt::Long; my $interval = 5; my $verbose = 0; die "Unknown options" unless GetOptions( 'interval|n=i' => \$interval, 'verbose|v' => \$verbose ); my $quit_flag = 0; $SIG{TERM} = sub { $quit_flag = 1; }; our @EXPORT = qw(schwartz_decl schwartz_work schwartz_on_idle schwartz_on_afterwork schwartz_on_prework schwartz_prioritize); my $sclient; my $prioritize = 0; my $on_idle = sub { }; my $on_afterwork = sub { }; my $on_prework = sub { 1 }; # return 1 to proceed and do work my $used_role; sub schwartz_init { my ($role) = @_; $role ||= 'drain'; $sclient = LJ::theschwartz( { role => $role } ) or die "Could not get schwartz client"; $used_role = $role; # save success role $sclient->set_verbose($verbose); $sclient->set_prioritize($prioritize); } sub schwartz_decl { my ( $classname, $role ) = @_; $role ||= 'drain'; die "Already connected to TheSchwartz with role '$used_role'" if defined $used_role and $role ne $used_role; schwartz_init($role) unless $sclient; $sclient->can_do($classname); } sub schwartz_prioritize { $prioritize = $_[0] ? 1 : 0; $sclient->set_prioritize($prioritize) if $sclient; } sub schwartz_on_idle { my ($code) = @_; $on_idle = $code; } sub schwartz_on_afterwork { my ($code) = @_; $on_afterwork = $code; } # coderef to return 1 to proceed, 0 to sleep sub schwartz_on_prework { my ($code) = @_; $on_prework = $code; } sub schwartz_work { my $sleep = 0; schwartz_init() unless $sclient; LJ::Worker->setup_mother(); my $last_death_check = time(); while (1) { LJ::start_request(); LJ::Worker->check_limits(); # check to see if we should die my $now = time(); if ( $now != $last_death_check ) { $last_death_check = $now; if ( -e "$LJ::VAR/$$.please_die" ) { unlink "$LJ::VAR/$$.please_die"; # Using exit here since "die" would be caught by an eval block exit 1; } } my $did_work = 0; if ( $on_prework->() ) { $did_work = $sclient->work_once; $on_afterwork->($did_work); exit 0 if $quit_flag; } if ($did_work) { $sleep--; $sleep = 0 if $sleep < 0; } else { $on_idle->(); $sleep = $interval if ++$sleep > $interval; sleep $sleep; } # do request cleanup before we process another job LJ::end_request(); } } 1;