Skip to content

Commit c06e3ca

Browse files
authored
Merge pull request #8 from mitelg/add-php-82
Add PHP 8.2 compatibility
2 parents 568aae7 + 135410a commit c06e3ca

8 files changed

+129
-31
lines changed

.github/workflows/build.yaml

+33-1
Original file line numberDiff line numberDiff line change
@@ -10,16 +10,48 @@ on:
1010
- main
1111

1212
jobs:
13+
static-analysis:
14+
runs-on: ubuntu-latest
15+
steps:
16+
- name: Checkout
17+
uses: actions/checkout@v2
18+
19+
- name: Install PHP
20+
uses: shivammathur/setup-php@v2
21+
with:
22+
coverage: none
23+
php-version: 7.4
24+
25+
- name: Determine composer cache directory
26+
id: determine-composer-cache-directory
27+
run: echo "::set-output name=directory::$(composer config cache-dir)"
28+
29+
- name: Cache dependencies installed with composer
30+
uses: actions/cache@v2
31+
with:
32+
path: ${{ steps.determine-composer-cache-directory.outputs.directory }}
33+
key: php-7.4-composer-${{ hashFiles('composer.json') }}
34+
restore-keys: php-7.4-composer-
35+
36+
- name: Install dependencies
37+
run: composer install --no-interaction --no-progress
38+
39+
- name: Check code style
40+
run: composer cs-check
41+
42+
- name: Check PHPStan
43+
run: composer phpstan
44+
1345
unit-tests:
1446
name: Tests
1547
runs-on: ubuntu-latest
1648
strategy:
1749
matrix:
1850
php-version:
19-
- 7.3
2051
- 7.4
2152
- 8.0
2253
- 8.1
54+
- 8.2
2355
steps:
2456
- name: Checkout
2557
uses: actions/checkout@v2

.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -4,3 +4,5 @@ clover.xml
44
tests/testfile_*.txt
55
infection-log.txt
66
.phpunit.result.cache
7+
.idea
8+
.php-cs-fixer.cache

.php-cs-fixer.php

+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
use PhpCsFixer\Config;
6+
use PhpCsFixer\Finder;
7+
8+
$finder = Finder::create()
9+
->in('src')
10+
->in('tests');
11+
12+
$config = new Config();
13+
14+
return $config
15+
->setRiskyAllowed(true)
16+
->setRules([
17+
'@PSR12' => true,
18+
'@Symfony' => true,
19+
'concat_space' => ['spacing' => 'one'],
20+
'declare_strict_types' => true,
21+
'increment_style' => ['style' => 'post'],
22+
'native_function_invocation' => true,
23+
'no_superfluous_phpdoc_tags' => true,
24+
'phpdoc_summary' => false,
25+
'yoda_style' => ['equal' => false, 'identical' => false, 'less_and_greater' => false],
26+
])
27+
->setFinder($finder);

CHANGELOG.md

+7-3
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,17 @@
1+
## 1.3.0
2+
- Introduced support for PHP 8.2
3+
- Added code style fixer
4+
- Added PHPStan analysis
5+
- Drop support for PHP 7.3
6+
17
## 1.2.0
28
- Introduced support for PHP 8.1
39

410
## 1.1.0
511
- Introduced support for PHP 7.4
612
- Introduced support for PHP 8.0
713
- Bumped minimum PHP Version to 7.3
8-
- Introduced github actions
14+
- Introduced GitHub actions
915

1016
## 1.0.1
1117

@@ -29,5 +35,3 @@
2935
## 0.1.0
3036

3137
- Initial Release
32-
33-

composer.json

+14-3
Original file line numberDiff line numberDiff line change
@@ -10,11 +10,14 @@
1010
}
1111
],
1212
"require": {
13-
"php": "^7.3|^7.4|^8.0|^8.1"
13+
"php": "^7.4 || ^8.0"
1414
},
1515
"require-dev": {
1616
"phpunit/phpunit": "^9.4",
17-
"infection/infection": "^0.18"
17+
"infection/infection": "^0.26",
18+
"friendsofphp/php-cs-fixer": "3.13.2",
19+
"phpstan/phpstan": "1.9.14",
20+
"phpstan/phpstan-phpunit": "1.3.3"
1821
},
1922
"autoload": {
2023
"psr-4": {
@@ -27,6 +30,14 @@
2730
}
2831
},
2932
"scripts": {
30-
"test": "phpunit"
33+
"test": "phpunit",
34+
"cs-fix": "php-cs-fixer fix -v",
35+
"cs-check": "php-cs-fixer fix --dry-run -v",
36+
"phpstan": "phpstan"
37+
},
38+
"config": {
39+
"allow-plugins": {
40+
"infection/extension-installer": true
41+
}
3142
}
3243
}

phpstan.neon

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
parameters:
2+
level: max
3+
phpVersion: 70300
4+
paths:
5+
- src
6+
- tests

src/LineReader.php

+22-9
Original file line numberDiff line numberDiff line change
@@ -1,17 +1,21 @@
11
<?php
22

3+
declare(strict_types=1);
4+
35
namespace Bcremer\LineReader;
46

57
final class LineReader
68
{
79
/**
810
* Prevent instantiation
911
*/
10-
private function __construct() {}
12+
private function __construct()
13+
{
14+
}
1115

1216
/**
13-
* @param string $filePath
14-
* @return \Generator
17+
* @return \Generator<int, string, void, void>
18+
*
1519
* @throws \InvalidArgumentException if $filePath is not readable
1620
*/
1721
public static function readLines(string $filePath): \Generator
@@ -24,8 +28,8 @@ public static function readLines(string $filePath): \Generator
2428
}
2529

2630
/**
27-
* @param string $filePath
28-
* @return \Generator
31+
* @return \Generator<int, string, void, void>
32+
*
2933
* @throws \InvalidArgumentException if $filePath is not readable
3034
*/
3135
public static function readLinesBackwards(string $filePath): \Generator
@@ -35,13 +39,15 @@ public static function readLinesBackwards(string $filePath): \Generator
3539
}
3640

3741
$size = filesize($filePath);
42+
if (!\is_int($size)) {
43+
throw new \RuntimeException('Could not get file size');
44+
}
3845

3946
return self::readBackwards($fh, $size);
4047
}
4148

4249
/**
4350
* @param resource $fh
44-
* @return \Generator
4551
*/
4652
private static function read($fh): \Generator
4753
{
@@ -60,9 +66,10 @@ private static function read($fh): \Generator
6066
* a newline character.
6167
*
6268
* @see http://stackoverflow.com/a/10494801/147634
69+
*
6370
* @param resource $fh
64-
* @param int $pos
65-
* @return \Generator
71+
*
72+
* @return \Generator<int, string, void, void>
6673
*/
6774
private static function readBackwards($fh, int $pos): \Generator
6875
{
@@ -79,7 +86,7 @@ private static function readBackwards($fh, int $pos): \Generator
7986
continue;
8087
}
8188

82-
if ($pos === 0) {
89+
if ($pos === 0 && \is_array($buffer)) {
8390
yield array_pop($buffer);
8491
break;
8592
}
@@ -91,7 +98,13 @@ private static function readBackwards($fh, int $pos): \Generator
9198
$pos -= $bufferSize;
9299
}
93100
fseek($fh, $pos);
101+
if ($bufferSize < 0) {
102+
throw new \RuntimeException('Buffer size cannot be negative');
103+
}
94104
$chunk = fread($fh, $bufferSize);
105+
if (!\is_string($chunk)) {
106+
throw new \RuntimeException('Could not read file');
107+
}
95108
if ($buffer === null) {
96109
// remove single trailing newline, rtrim cannot be used here
97110
if (substr($chunk, -1) === "\n") {

tests/LineReaderTest.php

+18-15
Original file line numberDiff line numberDiff line change
@@ -1,24 +1,29 @@
11
<?php
2+
3+
declare(strict_types=1);
4+
25
namespace Bcremer\LineReaderTests;
36

47
use Bcremer\LineReader\LineReader;
58
use PHPUnit\Framework\TestCase;
69

710
class LineReaderTest extends TestCase
811
{
9-
private static $maxLines;
10-
private static $testFile;
12+
private static int $maxLines;
13+
14+
private static string $testFile;
1115

1216
public static function setUpBeforeClass(): void
1317
{
14-
self::$maxLines = (int)getenv('TEST_MAX_LINES') ?: 10000;
15-
self::$testFile = __DIR__.'/testfile_'.self::$maxLines.'.txt';
18+
self::$maxLines = (int) getenv('TEST_MAX_LINES') ?: 10000;
19+
self::$testFile = __DIR__ . '/testfile_' . self::$maxLines . '.txt';
1620

1721
if (is_file(self::$testFile)) {
1822
return;
1923
}
2024

2125
$fh = fopen(self::$testFile, 'w');
26+
self::assertIsResource($fh);
2227
for ($i = 1; $i <= self::$maxLines; $i++) {
2328
fwrite($fh, "Line $i\n");
2429
}
@@ -45,8 +50,6 @@ public function testReadsAllLines(): void
4550
{
4651
$result = LineReader::readLines(self::$testFile);
4752

48-
self::assertInstanceOf(\Traversable::class, $result);
49-
5053
$firstLine = 1;
5154
$lastLine = self::$maxLines;
5255
$lineCount = self::$maxLines;
@@ -60,7 +63,7 @@ public function testReadsLinesByStartline(): void
6063

6164
$firstLine = 51;
6265
$lastLine = self::$maxLines;
63-
$lineCount = self::$maxLines-50;
66+
$lineCount = self::$maxLines - 50;
6467
$this->assertLines($lineGenerator, $firstLine, $lastLine, $lineCount);
6568
}
6669

@@ -90,15 +93,15 @@ public function testReadsLinesBackwardsWithOffsetAndLimit(): void
9093
$lineGenerator = LineReader::readLinesBackwards(self::$testFile);
9194
$lineGenerator = new \LimitIterator($lineGenerator, 10, 50);
9295

93-
$firstLine = self::$maxLines-10;
94-
$lastLine = self::$maxLines-59;
96+
$firstLine = self::$maxLines - 10;
97+
$lastLine = self::$maxLines - 59;
9598
$lineCount = 50;
9699
$this->assertLines($lineGenerator, $firstLine, $lastLine, $lineCount);
97100
}
98101

99102
public function testEmptyFile(): void
100103
{
101-
$testFile = __DIR__.'/testfile_empty.txt';
104+
$testFile = __DIR__ . '/testfile_empty.txt';
102105
$content = '';
103106
file_put_contents($testFile, $content);
104107

@@ -111,7 +114,7 @@ public function testEmptyFile(): void
111114

112115
public function testFileWithLeadingAndTrailingNewlines(): void
113116
{
114-
$testFile = __DIR__.'/testfile_space.txt';
117+
$testFile = __DIR__ . '/testfile_space.txt';
115118

116119
$content = <<<CONTENT
117120
@@ -159,20 +162,20 @@ public function testFileWithLeadingAndTrailingNewlines(): void
159162
/**
160163
* Runs the generator and asserts on first, last and the total line count
161164
*
162-
* @param \Traversable $generator
165+
* @param \Traversable<string> $generator
163166
*/
164-
private function assertLines(\Traversable $generator, string $firstLine, int $lastLine, int $lineCount): void
167+
private function assertLines(\Traversable $generator, int $firstLine, int $lastLine, int $lineCount): void
165168
{
166169
$count = 0;
167170
$line = '';
168171
foreach ($generator as $line) {
169172
if ($count === 0) {
170-
self::assertSame("Line $firstLine", $line, 'Expect first line');
173+
self::assertSame(sprintf('Line %s', $firstLine), $line, 'Expect first line');
171174
}
172175
$count++;
173176
}
174177

175-
self::assertSame("Line $lastLine", $line, 'Expect last line');
178+
self::assertSame(sprintf('Line %s', $lastLine), $line, 'Expect last line');
176179
self::assertSame($lineCount, $count, 'Expect total line count');
177180
}
178181
}

0 commit comments

Comments
 (0)