QRCodeReaderTestAbstract.php 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. <?php
  2. /**
  3. * Class QRCodeReaderTestAbstract
  4. *
  5. * @created 17.01.2021
  6. * @author Smiley <smiley@chillerlan.net>
  7. * @copyright 2021 Smiley
  8. * @license MIT
  9. *
  10. * @noinspection PhpComposerExtensionStubsInspection
  11. */
  12. namespace chillerlan\QRCodeTest;
  13. use chillerlan\QRCode\{QRCode, QROptions};
  14. use chillerlan\QRCode\Common\{EccLevel, LuminanceSourceInterface, Mode, Version};
  15. use chillerlan\QRCode\Decoder\Decoder;
  16. use chillerlan\QRCode\Output\QRGdImagePNG;
  17. use chillerlan\Settings\SettingsContainerInterface;
  18. use PHPUnit\Framework\Attributes\{DataProvider, Group};
  19. use PHPUnit\Framework\TestCase;
  20. use Exception, Generator;
  21. use function array_map, defined, realpath, sprintf, str_repeat, substr;
  22. /**
  23. * Tests the QR Code reader
  24. */
  25. abstract class QRCodeReaderTestAbstract extends TestCase{
  26. use QRMaxLengthTrait, QRMatrixDebugTrait;
  27. /** @see https://www.bobrosslipsum.com/ */
  28. protected const loremipsum = 'Just let this happen. We just let this flow right out of our minds. '
  29. .'Anyone can paint. We touch the canvas, the canvas takes what it wants. From all of us here, '
  30. .'I want to wish you happy painting and God bless, my friends. A tree cannot be straight if it has a crooked trunk. '
  31. .'You have to make almighty decisions when you\'re the creator. I guess that would be considered a UFO. '
  32. .'A big cotton ball in the sky. I\'m gonna add just a tiny little amount of Prussian Blue. '
  33. .'They say everything looks better with odd numbers of things. But sometimes I put even numbers—just '
  34. .'to upset the critics. We\'ll lay all these little funky little things in there. ';
  35. protected const samplesDir = __DIR__.'/samples/';
  36. protected SettingsContainerInterface|QROptions $options;
  37. protected function setUp():void{
  38. if(!defined('READER_TEST_MAX_VERSION')){
  39. $this::markTestSkipped('READER_TEST_MAX_VERSION not defined (see phpunit.xml.dist)');
  40. }
  41. $this->options = new QROptions;
  42. $this->options->readerUseImagickIfAvailable = false;
  43. }
  44. /**
  45. * @phpstan-return array<string, array{0: string, 1: string, 2: bool}>
  46. */
  47. public static function qrCodeProvider():array{
  48. return [
  49. 'helloworld' => ['hello_world.png', 'Hello world!', false],
  50. // covers mirroring
  51. 'mirrored' => ['hello_world_mirrored.png', 'Hello world!', false],
  52. // data modes
  53. 'byte' => ['byte.png', 'https://smiley.codes/qrcode/', true],
  54. 'numeric' => ['numeric.png', '123456789012345678901234567890', false],
  55. 'alphanum' => ['alphanum.png', 'ABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890 $%*+-./:', false],
  56. 'kanji' => ['kanji.png', '茗荷茗荷茗荷茗荷', false],
  57. // covers most of ReedSolomonDecoder
  58. 'damaged' => ['damaged.png', 'https://smiley.codes/qrcode/', false],
  59. // covers attempt to read 2nd (bottom left) version info
  60. 'version2' => ['damaged_version.png', 'https://www.youtube.com/watch?v=dQw4w9WgXcQ', false],
  61. // covers Binarizer::getHistogramBlackMatrix()
  62. 'smol' => ['smol.png', 'https://smiley.codes/qrcode/', false],
  63. // tilted 22° CCW
  64. 'tilted' => ['tilted.png', 'Hello world!', false],
  65. // rotated 90° CW
  66. 'rotated' => ['rotated.png', 'Hello world!', false],
  67. // color gradient (from old svg example)
  68. 'gradient' => ['example_svg.png', 'https://www.youtube.com/watch?v=DLzxrzFCyOs&t=43s', true],
  69. // color gradient (from svg example)
  70. 'dots' => ['example_svg_dots.png', 'https://www.youtube.com/watch?v=dQw4w9WgXcQ', true],
  71. ];
  72. }
  73. abstract protected function getLuminanceSourceFromFile(
  74. string $file,
  75. SettingsContainerInterface|QROptions $options
  76. ):LuminanceSourceInterface;
  77. #[Group('slow')]
  78. #[DataProvider('qrCodeProvider')]
  79. public function testReader(string $img, string $expected, bool $grayscale):void{
  80. if($grayscale){
  81. $this->options->readerGrayscale = true;
  82. $this->options->readerIncreaseContrast = true;
  83. }
  84. $luminanceSource = $this->getLuminanceSourceFromFile(realpath($this::samplesDir.$img), $this->options);
  85. $result = (new Decoder)->decode($luminanceSource);
  86. $this->debugMatrix($result->getQRMatrix());
  87. $this::assertSame($expected, (string)$result);
  88. }
  89. public function testReaderMultiMode():void{
  90. $this->options->outputInterface = QRGdImagePNG::class;
  91. $this->options->outputBase64 = false;
  92. $numeric = '123456789012345678901234567890';
  93. $alphanum = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890 $%*+-./:';
  94. $kanji = '漂う花の香り';
  95. $hanzi = '无可奈何燃花作香';
  96. $byte = 'https://smiley.codes/qrcode/';
  97. $qrcode = (new QRCode($this->options))
  98. ->addNumericSegment($numeric)
  99. ->addAlphaNumSegment($alphanum)
  100. ->addKanjiSegment($kanji)
  101. ->addHanziSegment($hanzi)
  102. ->addByteSegment($byte)
  103. ;
  104. $result = $qrcode->readFromBlob($qrcode->render());
  105. $this::assertSame($numeric.$alphanum.$kanji.$hanzi.$byte, $result->data);
  106. }
  107. public static function dataTestProvider():Generator{
  108. $str = str_repeat(self::loremipsum, 5);
  109. $eccLevels = array_map(fn(int $ecc):EccLevel => new EccLevel($ecc), [EccLevel::L, EccLevel::M, EccLevel::Q, EccLevel::H]);
  110. /**
  111. * @noinspection PhpUndefinedConstantInspection - see phpunit.xml.dist
  112. * @phpstan-ignore-next-line
  113. */
  114. for($v = 1; $v <= READER_TEST_MAX_VERSION; $v++){
  115. $version = new Version($v);
  116. foreach($eccLevels as $eccLevel){
  117. yield 'version: '.$version.$eccLevel => [
  118. $version,
  119. $eccLevel,
  120. substr($str, 0, (self::getMaxLengthForMode(Mode::BYTE, $version, $eccLevel) ?? '')),
  121. ];
  122. }
  123. }
  124. }
  125. #[Group('slow')]
  126. #[DataProvider('dataTestProvider')]
  127. public function testReadData(Version $version, EccLevel $ecc, string $expected):void{
  128. $this->options->outputInterface = QRGdImagePNG::class;
  129. $this->options->imageTransparent = false;
  130. $this->options->eccLevel = $ecc->getLevel();
  131. $this->options->version = $version->getVersionNumber();
  132. $this->options->outputBase64 = false;
  133. // what's interesting is that a smaller scale seems to produce fewer reader errors???
  134. // usually from version 20 up, independend of the luminance source
  135. // scale 1-2 produces none, scale 3: 1 error, scale 4: 6 errors, scale 5: 5 errors, scale 10: 10 errors
  136. // @see \chillerlan\QRCode\Detector\GridSampler::checkAndNudgePoints()
  137. $this->options->scale = 2;
  138. try{
  139. $qrcode = new QRCode($this->options);
  140. $imagedata = $qrcode->render($expected);
  141. $result = $qrcode->readFromBlob($imagedata);
  142. }
  143. catch(Exception $e){
  144. $this::markTestSkipped(sprintf('skipped version %s%s: %s', $version, $ecc, $e->getMessage()));
  145. }
  146. $this::assertSame($expected, $result->data);
  147. $this::assertSame($version->getVersionNumber(), $result->version->getVersionNumber());
  148. $this::assertSame($ecc->getLevel(), $result->eccLevel->getLevel());
  149. }
  150. }