App-SD
view release on metacpan or search on metacpan
lib/App/SD/Replica/github/PushEncoder.pm view on Meta::CPAN
package App::SD::Replica::github::PushEncoder;
use Any::Moose;
use Params::Validate;
use Path::Class;
has sync_source => (
isa => 'App::SD::Replica::github',
is => 'rw',
);
sub integrate_change {
my $self = shift;
my ( $change, $changeset ) = validate_pos(
@_,
{ isa => 'Prophet::Change' },
{ isa => 'Prophet::ChangeSet' }
);
my ( $id, $record );
# if the original_sequence_no of this changeset is <=
# the last changeset our sync source for the original_sequence_no, we can skip it.
# XXX TODO - this logic should be at the changeset level, not the cahnge level, as it applies to all
# changes in the changeset
# warn $self->sync_source->app_handle->handle->last_changeset_from_source(
# $changeset->original_source_uuid ), "\n";
return
if $self->sync_source->app_handle->handle->last_changeset_from_source(
$changeset->original_source_uuid ) >= $changeset->original_sequence_no;
my $before_integration = time();
eval {
if ( $change->record_type eq 'ticket'
and $change->change_type eq 'add_file' )
{
$id = $self->integrate_ticket_create( $change, $changeset );
$self->sync_source->record_remote_id_for_pushed_record(
uuid => $change->record_uuid,
remote_id => $id,
);
}
elsif ( $change->record_type eq 'comment'
and $change->change_type eq 'add_file' )
{
$id = $self->integrate_comment( $change, $changeset );
}
elsif ( $change->record_type eq 'ticket' ) {
$id = $self->integrate_ticket_update( $change, $changeset );
}
else {
$self->sync_source->log(
'I have no idea what I am doing for ' . $change->record_uuid );
return;
}
$self->sync_source->record_pushed_transactions(
start_time => $before_integration,
ticket => $id,
changeset => $changeset,
);
};
if ( my $err = $@ ) {
$self->sync_source->log( "Push error: " . $err );
}
return $id;
}
sub integrate_ticket_update {
my $self = shift;
my ( $change, $changeset ) = validate_pos(
@_,
{ isa => 'Prophet::Change' },
{ isa => 'Prophet::ChangeSet' }
);
# Figure out the remote site's ticket ID for this change's record
my $remote_ticket_id =
$self->sync_source->remote_id_for_uuid( $change->record_uuid );
my $ticket = $self->sync_source->github->issue();
my $attr = $self->_recode_props_for_integrate($change);
$ticket->edit( $remote_ticket_id, $attr->{title}, $attr->{body} );
if ( $attr->{status} ) {
$ticket->reopen( $remote_ticket_id ) if $attr->{status} eq 'open';
$ticket->close( $remote_ticket_id ) if $attr->{status} eq 'closed';
}
return $remote_ticket_id;
}
sub integrate_ticket_create {
my $self = shift;
my ( $change, $changeset ) = validate_pos(
@_,
{ isa => 'Prophet::Change' },
{ isa => 'Prophet::ChangeSet' }
);
# Build up a ticket object out of all the record's attributes
my $ticket = $self->sync_source->github->issue;
my $attr = $self->_recode_props_for_integrate($change);
my $new =
$ticket->open( $attr->{title}, $attr->{body} );
# TODO: better error handler?
if ( $new->{error} ) {
die "\n\n$new->{error}";
}
return $new->{number};
}
sub integrate_comment {
my $self = shift;
my ( $change, $changeset ) = validate_pos(
@_,
{ isa => 'Prophet::Change' },
{ isa => 'Prophet::ChangeSet' }
);
# Figure out the remote site's ticket ID for this change's record
my %props = map { $_->name => $_->new_value } $change->prop_changes;
my $ticket_id = $self->sync_source->remote_id_for_uuid( $props{'ticket'} );
my $ticket = $self->sync_source->github->issue();
$ticket->comment($ticket_id, $props{'content'});
return $ticket_id;
}
sub _recode_props_for_integrate {
my $self = shift;
my ($change) = validate_pos( @_, { isa => 'Prophet::Change' } );
my %props = map { $_->name => $_->new_value } $change->prop_changes;
my %attr;
for my $key ( keys %props ) {
if ( $key eq 'summary' ) {
$attr{title} = $props{$key};
}
elsif ( $key eq 'body' ) {
$attr{$key} = $props{$key};
}
elsif ( $key eq 'status' ) {
$attr{state} = $props{$key} =~ /new|open/ ? 'open' : 'closed';
}
}
return \%attr;
}
__PACKAGE__->meta->make_immutable;
no Any::Moose;
1;
( run in 0.610 second using v1.01-cache-2.11-cpan-39bf76dae61 )