Compare commits

..

8 commits
v1.4 ... main

14 changed files with 242 additions and 171 deletions

14
.github/workflows/phpstan.yml vendored Normal file
View file

@ -0,0 +1,14 @@
name: phpstan
on: push
jobs:
phpstan:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- uses: php-actions/composer@v6
with:
dev: yes
- uses: php-actions/phpstan@v3
with:
memory_limit: 1G
args: --memory-limit 1G

11
.github/workflows/phpunit.yml vendored Normal file
View file

@ -0,0 +1,11 @@
name: phpunit
on: push
jobs:
phpunit:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- uses: php-actions/composer@v6
with:
dev: yes
- uses: php-actions/phpunit@v3

View file

@ -1,6 +1,7 @@
# Flatrr
[![Build Status](https://travis-ci.org/jobyone/flatrr.svg?branch=main)](https://travis-ci.org/jobyone/flatrr)
[![phpstan](https://github.com/jobyone/flatrr/actions/workflows/phpstan.yml/badge.svg?branch=v1.5)](https://github.com/jobyone/flatrr/actions/workflows/phpstan.yml)
[![phpunit](https://github.com/jobyone/flatrr/actions/workflows/phpunit.yml/badge.svg?branch=v1.5)](https://github.com/jobyone/flatrr/actions/workflows/phpunit.yml)
[![Latest Stable Version](http://poser.pugx.org/byjoby/flatrr/v)](https://packagist.org/packages/byjoby/flatrr)
[![Total Downloads](http://poser.pugx.org/byjoby/flatrr/downloads)](https://packagist.org/packages/byjoby/flatrr)
[![Latest Unstable Version](http://poser.pugx.org/byjoby/flatrr/v/unstable)](https://packagist.org/packages/byjoby/flatrr)

View file

@ -3,12 +3,16 @@
"description": "A library for working with multi-dimensional arrays through flattened keys",
"type": "library",
"license": "MIT",
"authors": [{
"authors": [
{
"name": "Joby Elliott",
"email": "joby@byjoby.com"
}],
}
],
"require-dev": {
"phpunit/phpunit": "^9.5"
"phpunit/phpunit": "^9.5",
"phpstan/phpstan": "^1.9",
"squizlabs/php_codesniffer": "^3.7"
},
"autoload": {
"psr-4": {
@ -21,12 +25,12 @@
}
},
"scripts": {
"test": [
"phpunit"
]
"test": "phpunit",
"stan": "phpstan",
"sniff": "phpcs"
},
"require": {
"php": ">=7.1",
"mustangostang/spyc": "^0.6.3"
"php": ">=8.1",
"symfony/yaml": "^6.4"
}
}

19
phpcs.xml Normal file
View file

@ -0,0 +1,19 @@
<?xml version="1.0"?>
<ruleset xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" name="PHP_CodeSniffer"
xsi:noNamespaceSchemaLocation="./vendor/squizlabs/php_codesniffer/phpcs.xsd">
<description>Coding Standard</description>
<file>src</file>
<arg name="basepath" value="." />
<arg name="colors" />
<arg name="parallel" value="75" />
<arg value="np" />
<rule ref="PSR12">
<exclude name="Generic.Files.LineEndings" />
<exclude name="Generic.NamingConventions.CamelCapsFunctionName" />
<exclude name="PSR1.Methods.CamelCapsMethodName" />
</rule>
</ruleset>

4
phpstan.neon Normal file
View file

@ -0,0 +1,4 @@
parameters:
level: 7
paths:
- src

View file

@ -1,7 +1,31 @@
<phpunit bootstrap="vendor/autoload.php">
<?xml version="1.0" encoding="UTF-8"?>
<phpunit
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:noNamespaceSchemaLocation="vendor/phpunit/phpunit/phpunit.xsd"
colors="true"
executionOrder="random"
failOnWarning="true"
failOnRisky="true"
failOnEmptyTestSuite="true"
beStrictAboutOutputDuringTests="true"
verbose="true"
bootstrap="vendor/autoload.php">
<php>
<ini name="display_errors" value="On" />
<ini name="error_reporting" value="-1" />
<ini name="xdebug.mode" value="coverage" />
</php>
<testsuites>
<testsuite name="All">
<directory>tests</directory>
<testsuite name="Tests">
<directory>tests/</directory>
</testsuite>
</testsuites>
<coverage>
<include>
<directory suffix=".php">src</directory>
</include>
<report>
<html outputDirectory="coverage" lowUpperBound="50" highLowerBound="90" />
</report>
</coverage>
</phpunit>

View file

@ -1,105 +1,81 @@
<?php
/* Flatrr | https://gitlab.com/byjoby/flatrr | MIT License */
/* Flatrr | https://github.com/jobyone/flatrr | MIT License */
namespace Flatrr\Config;
use Flatrr\SelfReferencingFlatArray;
use Spyc;
use Symfony\Component\Yaml\Yaml;
class Config extends SelfReferencingFlatArray implements ConfigInterface
{
public $strict = false;
public function readDir($dir, string $name = null, bool $overwrite = false)
public function readDir(string $dir, string $name = null, bool $overwrite = false): static
{
$dir = realpath($dir);
if (!$dir || !is_dir($dir)) {
return;
}
foreach (glob("$dir/*") as $f) {
if ($dir && is_dir($dir)) {
$glob = glob("$dir/*");
if ($glob) {
foreach ($glob as $f) {
if (is_file($f)) {
$this->readFile($f, $name, $overwrite);
}
}
}
}
return $this;
}
protected function parse(string $input, string $format): array
public function json(bool $raw = false): string
{
$fn = 'parse_' . $format;
if (!method_exists($this, $fn)) {
if ($this->strict) {
throw new \Exception("Don't know how to parse the format \"$format\"");
} else {
return null;
}
}
if ($out = $this->$fn($input)) {
return $out;
}
return array();
return json_encode($this->get(null, $raw), JSON_PRETTY_PRINT); // @phpstan-ignore-line
}
protected function parse_yaml($input)
public function yaml(bool $raw = false): string
{
return Spyc::YAMLLoadString($input);
return Yaml::dump(
$this->get(null, $raw),
2,
2,
Yaml::DUMP_MULTI_LINE_LITERAL_BLOCK
);
}
public function json($raw = false): string
{
return json_encode($this->get(null, $raw), JSON_PRETTY_PRINT);
}
public function yaml($raw = false): string
{
return Spyc::YAMLDump($this->get(null, $raw), 2);
}
protected function read_ini($filename)
/** @return array<mixed|mixed> */
protected function read_ini(string $filename): false|array
{
return parse_ini_file($filename, true);
}
protected function read_json($filename)
/** @return array<mixed|mixed> */
protected function read_json(string $filename): null|array
{
return json_decode(file_get_contents($filename), true);
/** @var string */
$data = file_get_contents($filename);
return json_decode($data, true);
}
protected function read_yaml($filename)
/** @return array<mixed|mixed> */
protected function read_yaml(string $filename): array
{
return Spyc::YAMLLoad($filename);
return Yaml::parseFile($filename);
}
protected function read_yml($filename)
/** @return array<mixed|mixed> */
protected function read_yml(string $filename): array
{
return $this->read_yaml($filename);
}
public function readFile($filename, string $name = null, bool $overwrite = false)
public function readFile(string $filename, string $name = null, bool $overwrite = false): static
{
if (!is_file($filename) || !is_readable($filename)) {
if ($this->strict) {
throw new \Exception("Couldn't read config file \"$filename\"");
} else {
return null;
}
}
$format = strtolower(preg_replace('/.+\./', '', $filename));
$fn = 'read_' . $format;
if (!method_exists($this, $fn)) {
if ($this->strict) {
throw new \Exception("Don't know how to read the format \"$format\"");
} else {
return null;
}
}
if (is_file($filename) && is_readable($filename) && method_exists($this, $fn)) {
$data = $this->$fn($filename);
if (!$data) {
if ($this->strict) {
throw new \Exception("Error reading \"" . $filename . "\"");
} else {
return null;
}
}
if ($data !== null) {
$this->merge($data, $name, $overwrite);
}
}
return $this;
}
}

View file

@ -1,12 +1,17 @@
<?php
/* Flatrr | https://gitlab.com/byjoby/flatrr | MIT License */
/* Flatrr | https://github.com/jobyone/flatrr | MIT License */
namespace Flatrr\Config;
interface ConfigInterface extends \ArrayAccess
use Flatrr\FlatArrayInterface;
interface ConfigInterface extends FlatArrayInterface
{
public function readFile($filename, string $name = null, bool $overwrite = false);
public function json($raw = false) : string;
public function yaml($raw = false) : string;
public function get(string $name = null, bool $raw = false);
public function merge($value, string $name = null, bool $overwrite = false);
public function readDir(string $dir, string $name = null, bool $overwrite = false): static;
public function readFile(string $filename, string $name = null, bool $overwrite = false): static;
public function json(bool $raw = false): string;
public function yaml(bool $raw = false): string;
public function get(null|string $name = null, bool $raw = false): mixed;
public function merge(mixed $value, string $name = null, bool $overwrite = false): static;
}

View file

@ -1,12 +1,18 @@
<?php
/* Flatrr | https://gitlab.com/byjoby/flatrr | MIT License */
/* Flatrr | https://github.com/jobyone/flatrr | MIT License */
namespace Flatrr;
class FlatArray implements FlatArrayInterface
{
use FlatArrayTrait;
public function __construct(array $data = null)
/**
* @param null|array<string|mixed> $data
* @return void
*/
public function __construct(null|array $data = null)
{
$this->merge($data);
}

View file

@ -1,16 +1,25 @@
<?php
/* Flatrr | https://gitlab.com/byjoby/flatrr | MIT License */
/* Flatrr | https://github.com/jobyone/flatrr | MIT License */
namespace Flatrr;
interface FlatArrayInterface extends \ArrayAccess, \Iterator
{
public function set(?string $name, $value);
public function get(?string $name = null);
public function unset(?string $name);
public function merge($value, string $name = null, bool $overwrite = false);
use ArrayAccess;
use Iterator;
public function push(?string $name, $value);
public function pop(?string $name);
public function unshift(?string $name, $value);
public function shift(?string $name);
/**
* @extends ArrayAccess<string,mixed>
* @extends Iterator<string,mixed>
*/
interface FlatArrayInterface extends ArrayAccess, Iterator
{
public function set(null|string $name, mixed $value): mixed;
public function get(null|string $name = null): mixed;
public function unset(null|string $name): static;
public function merge(mixed $value, string $name = null, bool $overwrite = false): static;
public function push(null|string $name, mixed $value): static;
public function pop(null|string $name): mixed;
public function unshift(null|string $name, mixed $value): static;
public function shift(null|string $name): mixed;
}

View file

@ -1,31 +1,35 @@
<?php
/* Flatrr | https://gitlab.com/byjoby/flatrr | MIT License */
/* Flatrr | https://github.com/jobyone/flatrr | MIT License */
namespace Flatrr;
trait FlatArrayTrait
{
private $_arrayData = array();
private $_flattenCache = array();
/** @var array<string|mixed> */
protected $_arrayData = [];
/** @var array<string|mixed> */
protected $_flattenCache = [];
public function push(?string $name, $value)
public function push(null|string $name, mixed $value): static
{
$arr = $this->flattenSearch($name);
if ($arr !== null && !is_array($arr)) {
return;
return $this;
}
if ($arr === null) {
$arr = [];
}
$arr[] = $value;
$this->set($name, $arr);
return $this;
}
public function pop(?string $name)
public function pop(null|string $name): mixed
{
$arr = $this->flattenSearch($name);
if ($arr !== null && !is_array($arr)) {
return;
return null;
}
$out = array_pop($arr);
$this->unset($name);
@ -33,24 +37,25 @@ trait FlatArrayTrait
return $out;
}
public function unshift(?string $name, $value)
public function unshift(null|string $name, mixed $value): static
{
$arr = $this->flattenSearch($name);
if ($arr !== null && !is_array($arr)) {
return;
return $this;
}
if ($arr === null) {
$arr = [];
}
array_unshift($arr, $value);
$this->set($name, $arr);
return $this;
}
public function shift(?string $name)
public function shift(null|string $name): mixed
{
$arr = $this->flattenSearch($name);
if ($arr !== null && !is_array($arr)) {
return;
return null;
}
$out = array_shift($arr);
$this->unset($name);
@ -58,62 +63,64 @@ trait FlatArrayTrait
return $out;
}
public function set(?string $name, $value)
public function set(null|string $name, mixed $value): static
{
return $this->flattenSearch($name, $value);
$this->flattenSearch($name, $value);
return $this;
}
public function get(?string $name = null)
public function get(null|string $name = null): mixed
{
return $this->flattenSearch($name);
}
function unset(?string $name)
public function unset(null|string $name): static
{
$this->flattenSearch($name, null, true);
return $this;
}
public function offsetSet($name, $value)
public function offsetSet($name, $value): void
{
return $this->set($name, $value);
$this->set($name, $value);
}
public function offsetGet($name)
public function offsetGet($name): mixed
{
return $this->get($name);
}
public function offsetExists($name)
public function offsetExists($name): bool
{
return $this->flattenSearch($name) !== null;
}
public function offsetUnset($name)
public function offsetUnset($name): void
{
$this->unset($name);
}
public function rewind()
public function rewind(): void
{
return reset($this->_arrayData);
reset($this->_arrayData);
}
public function current()
public function current(): mixed
{
return current($this->_arrayData);
}
public function next()
public function next(): void
{
return next($this->_arrayData);
next($this->_arrayData);
}
public function key()
public function key(): null|string|int
{
return key($this->_arrayData);
}
public function valid()
public function valid(): bool
{
return isset($this->_arrayData[$this->key()]);
}
@ -122,12 +129,11 @@ trait FlatArrayTrait
* Recursively set a value, with control over whether existing values or new
* values take precedence
*/
public function merge($value, string $name = null, bool $overwrite = false)
public function merge(mixed $value, string $name = null, bool $overwrite = false): static
{
if (!isset($this[$name])) {
//easiest possible outcome, old value doesn't exist, so we can just write the value
$this->set($name, $value);
return;
} elseif (is_array($value) && is_array($this->flattenSearch($name))) {
//both new and old values are arrays
foreach ($value as $k => $v) {
@ -136,14 +142,13 @@ trait FlatArrayTrait
}
$this->merge($v, $k, $overwrite);
}
return;
} else {
//old and new values exist, and one or both are not arrays, $overwrite rules the day
if ($overwrite) {
$this->set($name, $value);
}
return;
}
return $this;
}
/**
@ -151,10 +156,10 @@ trait FlatArrayTrait
* string. It sets it if $value exists, otherwise it returns the value if it
* exists.
*/
protected function flattenSearch(?string $name, $value = null, $unset = false)
protected function flattenSearch(null|string $name, mixed $value = null, bool $unset = false): mixed
{
if ($value !== null || $unset) {
$this->_flattenCache = array();
$this->_flattenCache = [];
}
if (!isset($this->_flattenCache[$name])) {
$this->_flattenCache[$name] = $this->doFlattenSearch($name, $value, $unset);
@ -162,7 +167,7 @@ trait FlatArrayTrait
return $this->_flattenCache[$name];
}
protected function doFlattenSearch(?string $name, $value = null, $unset = false)
protected function doFlattenSearch(null|string $name, mixed $value = null, bool $unset = false): mixed
{
//check for home strings
if ($name == '' || $name === null) {
@ -180,7 +185,7 @@ trait FlatArrayTrait
if ($value !== null) {
foreach ($name as $part) {
if (!isset($parent[$part])) {
$parent[$part] = array();
$parent[$part] = [];
}
$parent = &$parent[$part];
}
@ -201,9 +206,9 @@ trait FlatArrayTrait
//both value and destination are arrays, merge them
$parent[$key] = array_replace_recursive($parent[$key], $value);
} else {
//set the hard way
if (!is_array($parent)) {
$parent = array();
//destination is not an array, to set this we must overwrite it with an empty array
if (!is_array(@$parent[$key])) {
$parent[$key] = [];
}
$parent[$key] = $value;
}

View file

@ -1,13 +1,15 @@
<?php
/* Flatrr | https://gitlab.com/byjoby/flatrr | MIT License */
/* Flatrr | https://github.com/jobyone/flatrr | MIT License */
namespace Flatrr;
class SelfReferencingFlatArray extends FlatArray
{
/** @var array<string,string> */
protected $cache = [];
public function get(string $name = null, bool $raw = false, $unescape = true)
public function get(string $name = null, bool $raw = false, bool $unescape = true): mixed
{
$out = parent::get($name);
if ($raw) {
@ -20,48 +22,29 @@ class SelfReferencingFlatArray extends FlatArray
return $out;
}
public function set(?string $name, $value)
public function set(null|string $name, mixed $value): static
{
$this->cache = [];
return $this->filter(parent::set($name, $value));
$this->filter(parent::set($name, $value));
return $this;
}
public function push(?string $name, $value)
{
return $this->filter(parent::push($name, $value));
}
public function pop(?string $name)
public function pop(null|string $name): mixed
{
return $this->filter(parent::pop($name));
}
public function unshift(?string $name, $value)
{
return $this->filter(parent::unshift($name, $value));
}
public function shift(?string $name)
public function shift(null|string $name): mixed
{
return $this->filter(parent::shift($name));
}
public function rewind()
{
return $this->filter(parent::rewind());
}
public function next()
{
return $this->filter(parent::next());
}
public function current()
public function current(): mixed
{
return $this->filter(parent::current());
}
protected function unescape($value)
protected function unescape(mixed $value): mixed
{
//map this function onto array values
if (is_array($value)) {
@ -88,7 +71,7 @@ class SelfReferencingFlatArray extends FlatArray
/**
* Recursively replace ${var/name} type strings in string values with
*/
protected function filter($value)
protected function filter(mixed $value): mixed
{
//map this function onto array values
if (is_array($value)) {
@ -114,9 +97,14 @@ class SelfReferencingFlatArray extends FlatArray
return $value;
}
protected function filter_regex($matches)
/**
* @param array<int,null|string> $matches
* @return string
*/
protected function filter_regex(array $matches): string
{
if (null !== $value = $this->get($matches[1], false, false)) {
$value = $this->get($matches[1], false, false);
if ($value !== null) {
if (!is_array($value)) {
return $value;
}

View file

@ -6,7 +6,7 @@ declare(strict_types=1);
namespace Flatrr\Config;
use PHPUnit\Framework\TestCase;
use Spyc;
use Symfony\Component\Yaml\Yaml;
class ConfigTest extends TestCase
{
@ -38,6 +38,11 @@ class ConfigTest extends TestCase
$a = new Config();
$a->readFile(__DIR__ . '/configtest.yaml');
$this->assertEquals($data, $a->get());
//nonexistant files
$a = new Config();
$a->readFile(__DIR__ . '/does-not-exist.json');
$a->readFile(__DIR__ . '/does-not-exist.yaml');
$this->assertEquals([], $a->get());
}
public function testSerializing()
@ -52,7 +57,7 @@ class ConfigTest extends TestCase
//json
$this->assertEquals($data, json_decode($c->json(), true));
//yaml
$this->assertEquals($data, Spyc::YAMLLoad($c->yaml()));
$this->assertEquals($data, Yaml::parse($c->yaml()));
}
public function testReadingDirectory()