mourningdove/cgi-bin/DW/Task/XPost.pm
2026-05-24 01:03:05 +00:00

149 lines
4.6 KiB
Perl

#!/usr/bin/perl
#
# DW::Task::XPost
#
# SQS worker for crossposting entries to external accounts.
#
# Authors:
# Allen Petersen
# Mark Smith <mark@dreamwidth.org>
#
# Copyright (c) 2009-2026 by Dreamwidth Studios, LLC.
#
# This program is free software; you may redistribute it and/or modify it under
# the same terms as Perl itself. For a copy of the license, please reference
# 'perldoc perlartistic' or 'perldoc perlgpl'.
#
package DW::Task::XPost;
use strict;
use v5.10;
use Log::Log4perl;
my $log = Log::Log4perl->get_logger(__PACKAGE__);
use Time::HiRes qw/ gettimeofday /;
use DW::External::Account;
use DW::TaskQueue;
use LJ::Event::XPostSuccess;
use LJ::Lang;
use LJ::Protocol;
use LJ::User;
use base 'DW::Task';
sub work {
my ( $self, $handle ) = @_;
my $arg = { %{ $self->args->[0] } };
my ( $uid, $ditemid, $acctid, $password, $auth_challenge, $auth_response, $delete ) =
map { delete $arg->{$_} }
qw( uid ditemid accountid password auth_challenge
auth_response delete );
if ( keys %$arg ) {
$log->error( "Unknown keys: " . join( ", ", keys %$arg ) );
return DW::Task::COMPLETED;
}
unless ( defined $uid && defined $ditemid && defined $acctid ) {
$log->error("Missing argument");
return DW::Task::COMPLETED;
}
my $u = LJ::want_user($uid);
unless ($u) {
$log->error("Unable to load user with uid $uid");
return DW::Task::FAILED;
}
if ( $u->is_suspended ) {
$log->error("User is suspended.");
return DW::Task::COMPLETED;
}
my $acct = DW::External::Account->get_external_account( $u, $acctid );
unless ($acct) {
$log->error("Unable to load account $acctid for uid $uid");
return DW::Task::FAILED;
}
my $notify_fail = sub {
DW::TaskQueue->dispatch(
LJ::Event::XPostFailure->new(
$u, $acctid, $ditemid, ( $_[0] || 'Unknown error message.' )
)
);
};
# LJRossia is temporarily broken, so skip - but we do want to notify
if ( $acct->externalsite && $acct->externalsite->{sitename} eq 'LJRossia' ) {
my $ljr_msg =
"Crossposts to LJRossia are disabled until the remote site fixes their XMLRPC protocol handler.";
$notify_fail->($ljr_msg);
return DW::Task::COMPLETED;
}
# LiveJournal has broken (or disabled) client posts - notify the user
if ( $acct->externalsite && $acct->externalsite->{sitename} eq 'LiveJournal' ) {
my $ljr_msg =
"Crossposting to LiveJournal is temporarily disabled due to LiveJournal refusing "
. "connections from us. Please see https://dw-maintenance.dreamwidth.org/86004.html for more details.";
$notify_fail->($ljr_msg);
return DW::Task::COMPLETED;
}
my $domain = $acct->externalsite ? $acct->externalsite->{domain} : 'unknown';
my $entry = LJ::Entry->new( $u, ditemid => $ditemid );
unless ( defined $entry && ( $delete || $entry->valid ) ) {
$log->error("Unable to load entry $ditemid for uid $uid");
return DW::Task::FAILED;
}
my %auth;
if ($auth_response) {
%auth = ( 'auth_challenge' => $auth_challenge, 'auth_response' => $auth_response );
}
else {
%auth = ( 'password' => $password );
}
my $start = [gettimeofday];
my $result =
$delete ? $acct->delete_entry( \%auth, $entry ) : $acct->crosspost( \%auth, $entry );
if ( $result->{success} ) {
DW::TaskQueue->dispatch( LJ::Event::XPostSuccess->new( $u, $acctid, $ditemid ) );
DW::Stats::increment( 'dw.worker.crosspost.success', 1, ["domain:$domain"] );
$log->info( sprintf( "Successful post to %s for %s(%d).", $domain, $u->user, $u->id ) );
}
else {
# In case this was a connection timeout, then we want to do special
# handling to make sure we retry immediately
if ( $result->{error} =~ /Failed to connect/ ) {
$log->warn(
sprintf(
"Timeout posting to %s for %s(%d)... will retry.",
$domain, $u->user, $u->id
)
);
return DW::Task::FAILED;
}
# Some other failure, so let's just let it go through.
$notify_fail->( $result->{error} );
DW::Stats::increment( 'dw.worker.crosspost.failure', 1, ["domain:$domain"] );
$log->error(
sprintf(
"Failed to post to %s for %s(%d): %s.",
$domain, $u->user, $u->id, $result->{error} || 'Unknown error message.'
)
);
}
return DW::Task::COMPLETED;
}
1;