Skip to content

Commit 542e9ec

Browse files
fix linter issue
Signed-off-by: Thomas Poignant <[email protected]>
1 parent cb5035f commit 542e9ec

23 files changed

+640
-529
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
{"php":"8.0.30","version":"3.57.1","indent":" ","lineEnding":"\n","rules":{"binary_operator_spaces":{"default":"at_least_single_space"},"blank_line_after_opening_tag":true,"blank_line_between_import_groups":true,"blank_lines_before_namespace":true,"braces_position":{"allow_single_line_empty_anonymous_classes":true},"class_definition":{"inline_constructor_arguments":false,"space_before_parenthesis":true},"compact_nullable_type_declaration":true,"declare_equal_normalize":true,"lowercase_cast":true,"lowercase_static_reference":true,"new_with_parentheses":true,"no_blank_lines_after_class_opening":true,"no_leading_import_slash":true,"no_whitespace_in_blank_line":true,"ordered_class_elements":{"order":["use_trait"]},"ordered_imports":{"imports_order":["class","function","const"],"sort_algorithm":"none"},"return_type_declaration":true,"short_scalar_cast":true,"single_import_per_statement":{"group_to_single_imports":false},"single_trait_insert_per_statement":true,"ternary_operator_spaces":true,"unary_operator_spaces":{"only_dec_inc":true},"visibility_required":true,"blank_line_after_namespace":true,"constant_case":true,"control_structure_braces":true,"control_structure_continuation_position":true,"elseif":true,"function_declaration":true,"indentation_type":true,"line_ending":true,"lowercase_keywords":true,"method_argument_space":{"attribute_placement":"ignore","on_multiline":"ensure_fully_multiline"},"no_break_comment":true,"no_closing_tag":true,"no_multiple_statements_per_line":true,"no_space_around_double_colon":true,"no_spaces_after_function_name":true,"no_trailing_whitespace":true,"no_trailing_whitespace_in_comment":true,"single_blank_line_at_eof":true,"single_class_element_per_statement":{"elements":["property"]},"single_line_after_imports":true,"spaces_inside_parentheses":true,"statement_indentation":true,"switch_case_semicolon_to_colon":true,"switch_case_space":true,"encoding":true,"full_opening_tag":true},"hashes":{"tests\/unit\/controller\/OfrepApiTest.php":"ffc7e3ab94b4c58eb9b413fef1bef9de","tests\/unit\/GoFeatureFlagProviderTest.php":"062d381339c3bb95bd08b9fa05d1462a","tests\/TestCase.php":"efc0a05509f7101e640d29ec146c0aff","src\/util\/Validator.php":"527ab649217b08b455e0180e91817197","src\/config\/Config.php":"7a6642e67f641b823c4b1be1db0fc7d6","src\/controller\/OfrepApi.php":"bce6659287817694140d38bb23de0b60","src\/GoFeatureFlagProvider.php":"0535e5b9a10f7095225b332dadb6948b","src\/model\/OfrepApiResponse.php":"b34e7d13b252e0c6a722dbaefcbf2862","src\/exception\/RateLimitedException.php":"ac00860fc0c226c57774ddf246b7c5e0","src\/exception\/BaseGoffException.php":"18c57afd49dec676d9693c6bf3c014aa","src\/exception\/UnknownOfrepException.php":"0e429ddf9505f315e00bdb1991d4b0a9","src\/exception\/BaseOfrepException.php":"88dee34dc35e403219477ff521d7f225","src\/exception\/UnauthorizedException.php":"efb5188ddad635e8ee03bb887fa12fea","src\/exception\/InvalidConfigException.php":"34c2b8ba731cb801298ef08ddcb93f75","src\/exception\/ParseException.php":"f1a06dad98c1912857a72da31272a7d5","src\/exception\/InvalidContextException.php":"49da7c5d08ae5067d79fd053ab6adf7d","src\/exception\/FlagNotFoundException.php":"8910ff8e0f0ed12acbd5d414a5b12d5b","src\/util\/Mapper.php":"5e3e560cdad10c83da46e3e976f54bbb","src\/model\/OfrepApiSuccessResponse.php":"64ed4fbac2a5ebaa5765cc767b291c5f","src\/model\/OfrepApiErrorResponse.php":"ce96a809b1d4ec87b2794a9535a514d2"}}

providers/GoFeatureFlag/README.md

+2-1
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,7 @@ The constructor of the config object has the following options:
4444
| `endpoint` | **(mandatory)** The URL to access to the relay-proxy.<br />*(example: `https://relay.proxy.gofeatureflag.org/`)* |
4545
| `apiKey` | The token used to call the relay proxy. |
4646
| `customHeaders` | Any headers you want to add to call the relay-proxy. |
47+
| `httpclient` | The HTTP Client to use (if you want to use a custom one). _It has to be a `PSR-7` compliant implementation._ |
4748

4849
The only required option to create a `GoFeatureFlagProvider` is the URL _(`endpoint`)_ to your GO Feature Flag relay-proxy instance.
4950

@@ -62,7 +63,7 @@ $api->setProvider($provider);
6263
$client = $api->getClient();
6364
$evaluationContext = new MutableEvaluationContext(
6465
"214b796a-807b-4697-b3a3-42de0ec10a37",
65-
new Attributes(["email" => "[email protected]"])
66+
new Attributes(["email" => '[email protected]'])
6667
);
6768

6869
$value = $client->getBooleanDetails('integer_key', false, $evaluationContext);

providers/GoFeatureFlag/composer.json

+3-2
Original file line numberDiff line numberDiff line change
@@ -18,8 +18,9 @@
1818
],
1919
"require": {
2020
"php": "^8",
21+
"guzzlehttp/guzzle": "^7.9",
2122
"open-feature/sdk": "^2.0",
22-
"guzzlehttp/guzzle": "^7.9"
23+
"psr/http-message": "^2.0"
2324
},
2425
"require-dev": {
2526
"phpunit/phpunit": "^9",
@@ -68,7 +69,7 @@
6869
"@dev:analyze:phpstan",
6970
"@dev:analyze:psalm"
7071
],
71-
"dev:analyze:phpstan": "phpstan analyse --ansi --debug --memory-limit=512M",
72+
"dev:analyze:phpstan": "phpstan --ansi --debug --memory-limit=512M",
7273
"dev:analyze:psalm": "psalm",
7374
"dev:build:clean": "git clean -fX build/",
7475
"dev:lint": [

providers/GoFeatureFlag/phpstan.neon.dist

-3
Original file line numberDiff line numberDiff line change
@@ -3,9 +3,6 @@ parameters:
33
level: max
44
paths:
55
- ./src
6-
- ./tests
76
excludePaths:
87
- */tests/fixtures/*
98
- */tests/*/fixtures/*
10-
# TODO: Implement gRPC Completely
11-
- ./src/grpc

providers/GoFeatureFlag/src/GoFeatureFlagProvider.php

+41-19
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,14 @@
44

55
namespace OpenFeature\Providers\GoFeatureFlag;
66

7+
use DateTime;
8+
use OpenFeature\Providers\GoFeatureFlag\config\Config;
9+
use OpenFeature\Providers\GoFeatureFlag\controller\OfrepApi;
10+
use OpenFeature\Providers\GoFeatureFlag\exception\BaseOfrepException;
11+
use OpenFeature\Providers\GoFeatureFlag\exception\InvalidConfigException;
12+
use OpenFeature\Providers\GoFeatureFlag\exception\InvalidContextException;
13+
use OpenFeature\Providers\GoFeatureFlag\model\OfrepApiErrorResponse;
14+
use OpenFeature\Providers\GoFeatureFlag\util\Validator;
715
use OpenFeature\implementation\common\Metadata;
816
use OpenFeature\implementation\provider\AbstractProvider;
917
use OpenFeature\implementation\provider\ResolutionDetailsBuilder;
@@ -13,15 +21,15 @@
1321
use OpenFeature\interfaces\provider\Provider;
1422
use OpenFeature\interfaces\provider\Reason;
1523
use OpenFeature\interfaces\provider\ResolutionDetails;
16-
use OpenFeature\Providers\GoFeatureFlag\config\Config;
17-
use OpenFeature\Providers\GoFeatureFlag\controller\OfrepApi;
18-
use OpenFeature\Providers\GoFeatureFlag\exception\BaseOfrepException;
19-
use OpenFeature\Providers\GoFeatureFlag\exception\InvalidConfigException;
20-
use OpenFeature\Providers\GoFeatureFlag\util\Validator;
24+
use Throwable;
25+
26+
use function array_key_exists;
27+
use function gettype;
28+
use function implode;
2129

2230
class GoFeatureFlagProvider extends AbstractProvider implements Provider
2331
{
24-
protected static string $CLIENT_NAME = 'GO Feature Flag Provider';
32+
protected static string $NAME = 'GO Feature Flag Provider';
2533
private OfrepApi $ofrepApi;
2634

2735
/**
@@ -30,15 +38,15 @@ class GoFeatureFlagProvider extends AbstractProvider implements Provider
3038
public function __construct(Config $config)
3139
{
3240
Validator::validateConfig($config);
33-
if (is_array($config->getCustomHeaders()) && !array_key_exists("Content-Type", $config->getCustomHeaders())) {
34-
$config->getCustomHeaders()["Content-Type"] = "application/json";
41+
if (!array_key_exists('Content-Type', $config->getCustomHeaders())) {
42+
$config->addCustomHeader('Content-Type', 'application/json');
3543
}
3644
$this->ofrepApi = new OfrepApi($config);
3745
}
3846

3947
public function getMetadata(): Metadata
4048
{
41-
return new Metadata(self::$CLIENT_NAME);
49+
return new Metadata(static::$NAME);
4250
}
4351

4452
public function resolveBooleanValue(string $flagKey, bool $defaultValue, ?EvaluationContext $context = null): ResolutionDetails
@@ -47,25 +55,33 @@ public function resolveBooleanValue(string $flagKey, bool $defaultValue, ?Evalua
4755
}
4856

4957
/**
58+
* @param array<mixed>|array<string, mixed>|bool|DateTime|float|int|string|null $defaultValue
5059
* @param array<string> $allowedClasses
5160
*/
52-
private function evaluate(string $flagKey, mixed $defaultValue, array $allowedClasses, EvaluationContext $evaluationContext = null): ResolutionDetails
61+
private function evaluate(string $flagKey, array | string | bool | DateTime | float | int | null $defaultValue, array $allowedClasses, ?EvaluationContext $evaluationContext = null): ResolutionDetails
5362
{
5463
try {
55-
Validator::validateEvaluationContext($evaluationContext);
5664
Validator::validateFlagKey($flagKey);
5765

66+
if ($evaluationContext === null) {
67+
throw new InvalidContextException('Evaluation context is null');
68+
}
69+
if ($evaluationContext->getTargetingKey() === null || $evaluationContext->getTargetingKey() === '') {
70+
throw new InvalidContextException('Missing targetingKey in evaluation context');
71+
}
72+
5873
$apiResp = $this->ofrepApi->evaluate($flagKey, $evaluationContext);
5974

60-
if ($apiResp->isError()) {
75+
if ($apiResp instanceof OfrepApiErrorResponse) {
6176
$err = new ResolutionError(
62-
$apiResp->getErrorCode() ?? ErrorCode::GENERAL(),
63-
$apiResp->getErrorDetails()
77+
$apiResp->getErrorCode(),
78+
$apiResp->getErrorDetails(),
6479
);
80+
6581
return (new ResolutionDetailsBuilder())
6682
->withValue($defaultValue)
6783
->withError($err)
68-
->withReason(Reason::ERROR)
84+
->withReason($apiResp->getReason())
6985
->build();
7086
}
7187

@@ -74,39 +90,45 @@ private function evaluate(string $flagKey, mixed $defaultValue, array $allowedCl
7490
->withReason(Reason::ERROR)
7591
->withError(new ResolutionError(
7692
ErrorCode::TYPE_MISMATCH(),
77-
"Invalid type for $flagKey, got " . gettype($apiResp->getValue()) . " expected " . implode(", ", $allowedClasses)))
93+
"Invalid type for $flagKey, got " . gettype($apiResp->getValue()) . ' expected ' . implode(', ', $allowedClasses),
94+
))
7895
->withValue($defaultValue)
7996
->build();
8097
}
98+
8199
return (new ResolutionDetailsBuilder())
82100
->withValue($apiResp->getValue())
83101
->withReason($apiResp->getReason())
84102
->withVariant($apiResp->getVariant())
85103
->build();
86-
87104
} catch (BaseOfrepException $e) {
88105
$err = new ResolutionError($e->getErrorCode(), $e->getMessage());
106+
89107
return (new ResolutionDetailsBuilder())
90108
->withValue($defaultValue)
91109
->withError($err)
92110
->withReason(Reason::ERROR)
93111
->build();
94-
} catch (\Exception $e) {
112+
} catch (Throwable $e) {
95113
return (new ResolutionDetailsBuilder())
96114
->withValue($defaultValue)
97-
->withError(new ResolutionError(ErrorCode::GENERAL(), "An error occurred while evaluating the flag: " . $e->getMessage()))
115+
->withError(new ResolutionError(ErrorCode::GENERAL(), 'An error occurred while evaluating the flag: ' . $e->getMessage()))
98116
->withReason(Reason::ERROR)
99117
->build();
100118
}
101119
}
102120

121+
/**
122+
* @param array<string> $allowedClasses
123+
*/
103124
private function isValidType(mixed $value, array $allowedClasses): bool
104125
{
105126
foreach ($allowedClasses as $class) {
106127
if ($value instanceof $class || gettype($value) === $class) {
107128
return true;
108129
}
109130
}
131+
110132
return false;
111133
}
112134

providers/GoFeatureFlag/src/config/Config.php

+30-6
Original file line numberDiff line numberDiff line change
@@ -4,33 +4,57 @@
44

55
namespace OpenFeature\Providers\GoFeatureFlag\config;
66

7+
use Psr\Http\Client\ClientInterface;
8+
79
class Config
810
{
911
private string $endpoint;
12+
/**
13+
* @var array<string, string>
14+
*/
1015
private array $customHeaders = [];
1116

12-
public function __construct(string $endpoint, ?string $apiKey = '', ?array $custom_headers = [])
17+
/**
18+
* @var ClientInterface|null - The HTTP Client to use (if you want to use a custom one)
19+
*/
20+
private ?ClientInterface $httpclient;
21+
22+
/**
23+
* @param string $endpoint - The endpoint to your GO Feature Flag Instance
24+
* @param string|null $apiKey - API Key to use to connect to GO Feature Flag
25+
* @param array<string, string>|null $customHeaders - Custom headers you want to send
26+
* @param ClientInterface|null $httpclient - The HTTP Client to use (if you want to use a custom one)
27+
*/
28+
public function __construct(string $endpoint, ?string $apiKey = '', ?array $customHeaders = [], ?ClientInterface $httpclient = null)
1329
{
30+
$this->httpclient = $httpclient;
1431
$this->endpoint = $endpoint;
15-
$this->customHeaders = $custom_headers;
32+
$this->customHeaders = $customHeaders ?? [];
1633
if ($apiKey !== null && $apiKey !== '') {
1734
$this->customHeaders['Authorization'] = 'Bearer ' . $apiKey;
1835
}
1936
}
2037

21-
/**
22-
* @return string
23-
*/
2438
public function getEndpoint(): string
2539
{
2640
return $this->endpoint;
2741
}
2842

2943
/**
30-
* @return array
44+
* @return array<string, string>
3145
*/
3246
public function getCustomHeaders(): array
3347
{
3448
return $this->customHeaders;
3549
}
50+
51+
public function addCustomHeader(string $key, string $value): void
52+
{
53+
$this->customHeaders[$key] = $value;
54+
}
55+
56+
public function getHttpClient(): ?ClientInterface
57+
{
58+
return $this->httpclient;
59+
}
3660
}

0 commit comments

Comments
 (0)