docs/push.pl

200 lines
5.3 KiB
Perl
Executable File

#!/usr/bin/env perl
use strict;
use warnings;
use 5.010;
use open ':encoding(utf8)';
use File::Basename qw(fileparse);
use File::Temp;
use Getopt::Long;
use Mojo::UserAgent;
use Mojo::Util qw(decode encode slurp spurt trim);
use Term::UI;
use Term::ReadLine;
my $username;
my $password;
my $batchmode;
GetOptions(
'u|username=s' => \$username,
'p|password=s' => \$password,
'batchmode!' => \$batchmode,
) or die 'bad args';
die 'no repos specified' unless @ARGV;
my $ua = Mojo::UserAgent->new->max_redirects(10);
# Mojo::UserAgent::CookieJar::find is destructive...
# this is a nondestructive version that makes the login succeed on the Hub
Mojo::Util::monkey_patch 'Mojo::UserAgent::CookieJar', find => sub {
my ($self, $url) = @_;
return unless my $domain = my $host = $url->ihost;
my $path = $url->path->to_abs_string;
my @found;
while ($domain =~ /[^.]+\.[^.]+|localhost$/) {
next unless my $old = $self->{jar}{$domain};
# Grab cookies
#my $new = $self->{jar}{$domain} = [];
for my $cookie (@$old) {
next unless $cookie->domain || $host eq $cookie->origin;
# Check if cookie has expired
my $expires = $cookie->expires;
next if $expires && time > ($expires || 0);
#push @$new, $cookie;
# Taste cookie
next if $cookie->secure && $url->protocol ne 'https';
next unless Mojo::UserAgent::CookieJar::_path($cookie->path, $path);
my $name = $cookie->name;
my $value = $cookie->value;
push @found, Mojo::Cookie::Request->new(name => $name, value => $value);
}
}
# Remove another part
continue { $domain =~ s/^[^.]+\.?// }
return @found;
};
sub get_form_bits {
my $form = shift;
my $ret = {};
$form->find('input, textarea')->grep(sub {
!$_->match('input[type=submit], input[type=reset], input[type=button]')
&& defined($_->attr('name'))
})->each(sub {
my $e = shift;
my $name = $e->attr('name');
my $val;
if ($e->type eq 'textarea') {
$val = $e->text;
}
else {
$val = $e->attr('value');
}
$val = trim('' . ($val // ''));
$val =~ s!\r\n|\r!\n!g;
$ret->{$name} = $val;
});
return $ret;
}
my $term = Term::ReadLine->new('docker-library-docs-push');
sub prompt_for_edit {
my ($currentText, $proposedFile) = @_;
my $proposedText = slurp $proposedFile or warn 'missing ' . $proposedFile;
$proposedText = trim(decode('UTF-8', $proposedText));
return $currentText if $currentText eq $proposedText;
my @proposedFileBits = fileparse($proposedFile, qr!\.[^.]*!);
my $file = File::Temp->new(SUFFIX => $proposedFileBits[2]);
my $filename = $file->filename;
spurt encode('UTF-8', $currentText . "\n"), $filename;
system(qw(git --no-pager diff --no-index), $filename, $proposedFile);
my $reply;
if ($batchmode) {
$reply = 'yes';
}
else {
$reply = $term->get_reply(
prompt => 'Apply changes?',
choices => [ qw( yes vimdiff no quit ) ],
default => 'yes',
);
}
if ($reply eq 'quit') {
say 'quitting, as requested';
exit;
}
if ($reply eq 'yes') {
return $proposedText;
}
if ($reply eq 'vimdiff') {
system('vimdiff', $filename, $proposedFile) == 0 or die "vimdiff on $filename and $proposedFile failed";
return trim(decode('UTF-8', slurp($filename)));
}
return $currentText;
}
my $login = $ua->get('https://registry.hub.docker.com/account/login/');
die 'login failed' unless $login->success;
my $loginForm = $login->res->dom('#form-login')->first;
my $loginBits = get_form_bits($loginForm);
unless (defined $username) {
$username = $term->get_reply(prompt => 'Hub Username');
}
$loginBits->{username} = $username;
unless (defined $password) {
$password = $term->get_reply(prompt => 'Hub Password'); # TODO hide the input? O:)
}
$loginBits->{password} = $password;
$login = $ua->post($login->req->url->to_abs => {
Referer => $login->req->url->to_abs->to_string,
} => form => $loginBits);
die 'login failed' unless $login->success;
my $error = $login->res->dom('.alert-error');
if ($error->size) {
die $error->pluck('all_text')->join("\n") . "\n";
}
while (my $repo = shift) { # '/_/hylang', '/u/tianon/perl', etc
$repo =~ s!/+$!!;
$repo = '/_/' . $repo unless $repo =~ m!/!;
$repo = '/' . $repo unless $repo =~ m!^/!;
my $repoName = $repo;
$repoName =~ s!^.*/!!; # 'hylang', 'perl', etc
my $repoUrl = 'https://registry.hub.docker.com' . $repo . '/settings/';
my $repoTx = $ua->get($repoUrl);
warn 'failed to get: ' . $repoUrl and next unless $repoTx->success;
my $settingsForm = $repoTx->res->dom('form[name="repository_settings"]')->first;
die 'failed to find form on ' . $repoUrl unless $settingsForm;
my $settingsBits = get_form_bits($settingsForm);
my $hubShort = prompt_for_edit($settingsBits->{description}, $repoName . '/README-short.txt');
my $hubLong = prompt_for_edit($settingsBits->{full_description}, $repoName . '/README.md');
say 'no change to ' . $repoName . '; skipping' and next if $settingsBits->{description} eq $hubShort and $settingsBits->{full_description} eq $hubLong;
$settingsBits->{description} = $hubShort;
$settingsBits->{full_description} = $hubLong;
say 'updating ' . $repoName;
$repoTx = $ua->post($repoUrl => { Referer => $repoUrl } => form => $settingsBits);
die 'post to ' . $repoUrl . ' failed' unless $repoTx->success;
my $alert = $repoTx->res->dom('.alert-error');
if ($alert->size) {
my $text = trim $alert->pluck('all_text');
die 'update to ' . $repoUrl . ' failed:' . "\n" . $text if $text;
}
}