App-Sqitch
view release on metacpan or search on metacpan
lib/App/Sqitch/Command/checkout.pm view on Meta::CPAN
package App::Sqitch::Command::checkout;
use 5.010;
use strict;
use warnings;
use utf8;
use Moo;
use App::Sqitch::Types qw(Str);
use Locale::TextDomain qw(App-Sqitch);
use App::Sqitch::X qw(hurl);
use App::Sqitch::Plan;
use Path::Class qw(dir);
use Try::Tiny;
use namespace::autoclean;
extends 'App::Sqitch::Command';
with 'App::Sqitch::Role::RevertDeployCommand';
our $VERSION = 'v1.6.1'; # VERSION
has client => (
is => 'ro',
isa => Str,
lazy => 1,
default => sub {
my $sqitch = shift->sqitch;
return $sqitch->config->get( key => 'core.vcs.client' )
|| 'git' . ( App::Sqitch::ISWIN ? '.exe' : '' );
},
);
sub configure { {} }
sub execute {
my $self = shift;
my ($branch, $targets) = $self->parse_args(
target => $self->target,
names => [undef],
args => \@_,
no_changes => 1,
);
# Branch required.
$self->usage unless length $branch;
# Warn on multiple targets.
my $target = shift @{ $targets };
$self->warn(__x(
'Too many targets specified; connecting to {target}',
target => $target->name,
)) if @{ $targets };
# Now get to work.
my $sqitch = $self->sqitch;
my $git = $self->client;
my $engine = $target->engine;
$engine->with_verify( $self->verify );
$engine->log_only( $self->log_only );
$engine->lock_timeout( $self->lock_timeout );
# What branch are we on?
my $current_branch = $sqitch->probe($git, qw(rev-parse --abbrev-ref HEAD));
hurl {
ident => 'checkout',
message => __x('Already on branch {branch}', branch => $branch),
exitval => 1,
} if $current_branch eq $branch;
# Instantitate a plan without calling $target->plan.
my $from_plan = App::Sqitch::Plan->new(
sqitch => $sqitch,
target => $target,
);
# Load the branch plan from Git, assuming the same path.
my $to_plan = App::Sqitch::Plan->new(
sqitch => $sqitch,
target => $target,
)->parse(
# Git assumes a relative file name is relative to the repo root, even
# when you're in a subdirectory. So we have to prepend the currrent
# directory path ./ to convince it to read the file relative to the
# current directory. See #560 and
# https://git-scm.com/docs/gitrevisions#Documentation/gitrevisions.txt-emltrevgtltpathgtemegemHEADREADMEememmasterREADMEem
# for details.
# XXX Handle missing file/no contents.
scalar $sqitch->capture(
$git, 'show', "$branch:"
. File::Spec->catfile(File::Spec->curdir, $target->plan_file)
)
);
# Find the last change the plans have in common.
my $last_common_change;
for my $change ($to_plan->changes){
last unless $from_plan->get( $change->id );
$last_common_change = $change;
}
hurl checkout => __x(
'Branch {branch} has no changes in common with current branch {current}',
branch => $branch,
current => $current_branch,
) unless $last_common_change;
$sqitch->info(__x(
'Last change before the branches diverged: {last_change}',
last_change => $last_common_change->format_name_with_tags,
));
# Revert to the last common change.
$engine->set_variables( $self->_collect_revert_vars($target) );
$engine->plan( $from_plan );
try {
$engine->revert( $last_common_change->id, ! $self->no_prompt, $self->prompt_accept );
} catch {
# Rethrow unknown errors or errors with exitval > 1.
( run in 0.613 second using v1.01-cache-2.11-cpan-5a3173703d6 )