ETOOBUSY 🚀 minimal blogging for the impatient
QRate - complete program
TL;DR
The complete program for QRate… although still provisional.
The full program so far is available here:
qrate
3.93 KiB
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!