|  | @@ -0,0 +1,164 @@
 | 
	
		
			
				|  |  | +#!/usr/bin/perl -w
 | 
	
		
			
				|  |  | +use strict;
 | 
	
		
			
				|  |  | +use warnings;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +use File::Temp;
 | 
	
		
			
				|  |  | +use Getopt::Long;
 | 
	
		
			
				|  |  | +use Mojo::UserAgent;
 | 
	
		
			
				|  |  | +use Mojo::Util qw(slurp spurt trim);
 | 
	
		
			
				|  |  | +use Term::ReadKey;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +my $username;
 | 
	
		
			
				|  |  | +my $password;
 | 
	
		
			
				|  |  | +my $all;
 | 
	
		
			
				|  |  | +GetOptions(
 | 
	
		
			
				|  |  | +	'u|username=s' => \$username,
 | 
	
		
			
				|  |  | +	'p|password=s' => \$password,
 | 
	
		
			
				|  |  | +	'a|all' => \$all,
 | 
	
		
			
				|  |  | +) 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->epoch || 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, select, 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');
 | 
	
		
			
				|  |  | +		$ret->{$name} = '' . $e->val;
 | 
	
		
			
				|  |  | +		
 | 
	
		
			
				|  |  | +		if ($e->type eq 'textarea') {
 | 
	
		
			
				|  |  | +			$ret->{$name} = trim($ret->{$name});
 | 
	
		
			
				|  |  | +			$ret->{$name} =~ s!\r\n|\r!\n!g;
 | 
	
		
			
				|  |  | +		}
 | 
	
		
			
				|  |  | +	});
 | 
	
		
			
				|  |  | +	
 | 
	
		
			
				|  |  | +	return $ret;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +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) {
 | 
	
		
			
				|  |  | +	print 'Hub Username: ';
 | 
	
		
			
				|  |  | +	$username = ReadLine 0;
 | 
	
		
			
				|  |  | +	chomp $username;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +$loginBits->{username} = $username;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +unless (defined $password) {
 | 
	
		
			
				|  |  | +	print 'Hub Password: ';
 | 
	
		
			
				|  |  | +	ReadMode 2;
 | 
	
		
			
				|  |  | +	$password = ReadLine 0;
 | 
	
		
			
				|  |  | +	chomp $password;
 | 
	
		
			
				|  |  | +	ReadMode 0;
 | 
	
		
			
				|  |  | +	print "\n";
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +$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 = '/_/' . $repo unless $repo =~ m!/!;
 | 
	
		
			
				|  |  | +	$repo = '/' . $repo unless $repo =~ m!^/!;
 | 
	
		
			
				|  |  | +	$repo =~ s!/+$!!;
 | 
	
		
			
				|  |  | +	
 | 
	
		
			
				|  |  | +	my $repoName = $repo;
 | 
	
		
			
				|  |  | +	$repoName =~ s!^.*/!!; # 'hylang', 'perl', etc
 | 
	
		
			
				|  |  | +	
 | 
	
		
			
				|  |  | +	my $shortFile = $repoName . '/README-short.txt';
 | 
	
		
			
				|  |  | +	my $short = slurp $shortFile or die 'missing ' . $shortFile;
 | 
	
		
			
				|  |  | +	$short = trim $short;
 | 
	
		
			
				|  |  | +	
 | 
	
		
			
				|  |  | +	my $longFile = $repoName . '/README.md';
 | 
	
		
			
				|  |  | +	my $long = slurp $longFile or die 'missing ' . $longFile;
 | 
	
		
			
				|  |  | +	$long = trim $long;
 | 
	
		
			
				|  |  | +	
 | 
	
		
			
				|  |  | +	my $repoUrl = 'https://registry.hub.docker.com' . $repo . '/settings/';
 | 
	
		
			
				|  |  | +	my $repoTx = $ua->get($repoUrl);
 | 
	
		
			
				|  |  | +	die 'failed to get: ' . $repoUrl 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 = $settingsBits->{description};
 | 
	
		
			
				|  |  | +	my $hubLong = $settingsBits->{full_description};
 | 
	
		
			
				|  |  | +	
 | 
	
		
			
				|  |  | +	if ($hubShort ne $short) {
 | 
	
		
			
				|  |  | +		my $file = File::Temp->new(SUFFIX => '.txt');
 | 
	
		
			
				|  |  | +		my $filename = $file->filename;
 | 
	
		
			
				|  |  | +		spurt $hubShort . "\n", $filename;
 | 
	
		
			
				|  |  | +		system('vimdiff', $filename, $shortFile) == 0 or die "vimdiff on $filename and $shortFile failed";
 | 
	
		
			
				|  |  | +		$hubShort = trim(slurp($filename));
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +	
 | 
	
		
			
				|  |  | +	if ($hubLong ne $long) {
 | 
	
		
			
				|  |  | +		my $file = File::Temp->new(SUFFIX => '.md');
 | 
	
		
			
				|  |  | +		my $filename = $file->filename;
 | 
	
		
			
				|  |  | +		spurt $hubLong . "\n", $filename;
 | 
	
		
			
				|  |  | +		system('vimdiff', $filename, $longFile) == 0 or die "vimdiff on $filename and $longFile failed";
 | 
	
		
			
				|  |  | +		$hubLong = trim(slurp($filename));
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +	
 | 
	
		
			
				|  |  | +	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;
 | 
	
		
			
				|  |  | +	
 | 
	
		
			
				|  |  | +	$repoTx = $ua->post($repoUrl => { Referer => $repoUrl } => form => $settingsBits);
 | 
	
		
			
				|  |  | +	die 'post to ' . $repoUrl . ' failed' unless $repoTx->success;
 | 
	
		
			
				|  |  | +}
 |