QRate - complete program

TL;DR

The complete program for QRate… although still provisional.

The full program so far is available here:

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152
#!/usr/bin/env perl
use v5.24;
use warnings;
use experimental 'signatures';
no warnings 'experimental::signatures';

use FindBin '$Bin';
use lib "$Bin/local/lib/perl5";

use constant LIMIT => 1268;
use constant PAGE_SIZE => 'A4';
use constant X_PAGE_SIZE => 595;
use constant Y_PAGE_SIZE => 842;
use constant MIN_MARGIN => 30;
use constant BASE64_LINE_LENGTH => 76;
use constant LINES_PER_SLICE => 16;

use Imager::QRCode;
use Image::Magick;
use Barcode::ZBar;
use PDF::API2;
use Path::Tiny;
use Compress::Zlib qw< compress uncompress >;
use MIME::Base64 qw< encode_base64 decode_base64 >;

help_die() unless @ARGV == 3;
my ($command, $input, $output) = @ARGV;

if ($command eq 'encode') {
   encode($input, $output);
}
elsif ($command eq 'decode') {
   decode($input, $output);
}
else {
   help_die();
}

sub help_die {
   die "$0 <encode|decode> <input-file> <output-file>\n";
}

sub encode ($input, $output) {
   my $data = compress(path($input)->slurp_raw, 9) or die "compress()\n";
   assemble_pdf(qrcoder_it(slicer_it($data)))->save($output);
   return 0;
}

sub decode ($input, $output) {
   my $data = assemble_data(slice_reader_it(pdf_reader_it($input)));
   path($output)->spew_raw(uncompress($data));
   return 0;
}

sub assemble_pdf ($it) {
   my $x_margin = MIN_MARGIN;
   my $size = X_PAGE_SIZE - 2 * MIN_MARGIN;
   my $y_margin = int((Y_PAGE_SIZE - $size) / 2);
   my $pdf = PDF::API2->new;
   $pdf->default_page_size(PAGE_SIZE);
   while (my $png = $it->()) {
      print {*STDERR} '.';
      open my $fh, '<:raw', \$png or die "open(): $!\n";
      my $image = $pdf->image($fh);
      $pdf->page->object($image, $x_margin, $y_margin, $size, $size);
   }
   print {*STDERR} "\n";
   return $pdf;
}

sub qrcoder_it ($it) {
   my $qrcode = Imager::QRCode->new(
      size          => 8,
      margin        => 2,
      mode          => '8-bit',
      version       => 1,
      level         => 'H',
      casesensitive => 1,
      lightcolor    => Imager::Color->new(255, 255, 255),
      darkcolor     => Imager::Color->new(0, 0, 0),
   );
   return sub {
      my $data = $it->() // return;
      my $img = $qrcode->plot($data)->to_paletted;
      my $retval;
      $img->write(data => \$retval, type => 'png')
        or die "Failed to write: " . $img->errstr;
     return $retval;
   }
}

sub slicer_it ($data) {
   my @encoded = split m{\n}mxs, encode_base64($data);
   my $n_slice = 0;
   return sub {
      return unless @encoded;
      my @payload = splice @encoded, 0, LINES_PER_SLICE;
      my $header = $n_slice++ . (@encoded ? '+' : '.');
      return join "\n", $header, @payload, '';
   };
}

sub assemble_data ($it) {
   my @slices;
   while (my $slice = $it->()) {
      print {*STDERR} '.';
      push @slices, $slice;
   }
   print {*STDERR} "\n";
   @slices = sort { $a->{n} <=> $b->{n} } @slices;
   for my $n (0 .. $#slices) {
      die "missing slice $n\n" if $slices[$n]{n} != $n;
   }
   die "missing trailing slices\n" unless $slices[-1]{last};
   my $data = join '', map { $_->{data} } @slices;
   return decode_base64($data);
}

sub slice_reader_it ($it) {
   my $scanner = Barcode::ZBar::ImageScanner->new();
   $scanner->parse_config("enable");
   return sub {
      my $page = $it->() or return;

      my $image = Barcode::ZBar::Image->new();
      $image->set_format('Y800');
      $image->set_size($page->Get(qw(columns rows)));
      $image->set_data($page->ImageToBlob(magick => 'GRAY', depth => 8));

      my $n = $scanner->scan_image($image);

      return map {
         my ($header, $data) = split m{\n}mxs, $_->get_data, 2;
         my ($n, $more) = $header =~ m{\A (\d+) ([+.]) \z}mxs;
         return {
            n => $n,
            last => ($more eq '.' ? 1 : 0),
            data => $data,
         };
      } $image->get_symbols;
   }
}

sub pdf_reader_it ($file) {
   my $magick = Image::Magick->new();
   die if $magick->Read($file); # returns 0 on success
   my $n = 0;
   return sub {
      return if $n > $magick->$#*;
      return $magick->[$n++];
   };
}

Due to the several modules involved, I want to produce a proper Docker image and place it in a project on either GitLab or GitHub. We’ll see!

Thank you for your patience and stay safe!


Comments? Octodon, , GitHub, Reddit, or drop me a line!