|
2 | 2 |
|
3 | 3 | namespace LaunchDarkly\Tests\Impl\Evaluation; |
4 | 4 |
|
| 5 | +use LaunchDarkly\EvaluationReason; |
5 | 6 | use LaunchDarkly\Impl\Evaluation\Evaluator; |
6 | 7 | use LaunchDarkly\Impl\Evaluation\EvaluatorBucketing; |
7 | 8 | use LaunchDarkly\Impl\Model\Segment; |
@@ -53,7 +54,7 @@ public function testIncludedKeyForContextKind() |
53 | 54 | $this->assertTrue(self::segmentMatchesContext($segment, $multi)); |
54 | 55 | } |
55 | 56 |
|
56 | | - public function excludedKeyForContextKind() |
| 57 | + public function testExcludedKeyForContextKind() |
57 | 58 | { |
58 | 59 | $c1 = LDContext::create('key1', 'kind1'); |
59 | 60 | $c2 = LDContext::create('key2', 'kind2'); |
@@ -181,14 +182,87 @@ public function testNonMatchingRuleWithMultipleClauses() |
181 | 182 | $this->assertFalse(self::segmentMatchesContext($segment, $context)); |
182 | 183 | } |
183 | 184 |
|
| 185 | + public function recursionDepth() |
| 186 | + { |
| 187 | + return [[1], [2], [3], [4]]; |
| 188 | + } |
| 189 | + |
| 190 | + /** @dataProvider recursionDepth */ |
| 191 | + public function testSegmentReferencingSegment($depth) |
| 192 | + { |
| 193 | + $context = LDContext::create('foo'); |
| 194 | + |
| 195 | + $segmentKeys = []; |
| 196 | + for ($i = 0; $i < $depth; $i++) { |
| 197 | + $segmentKeys[] = "segmentkey$i"; |
| 198 | + } |
| 199 | + $flags = []; |
| 200 | + $requester = new MockFeatureRequester(); |
| 201 | + for ($i = 0; $i < $depth; $i++) { |
| 202 | + $builder = ModelBuilders::segmentBuilder($segmentKeys[$i]); |
| 203 | + if ($i == $depth - 1) { |
| 204 | + $builder->included($context->getKey()); |
| 205 | + } else { |
| 206 | + $builder->rule( |
| 207 | + ModelBuilders::segmentRuleBuilder() |
| 208 | + ->clause(ModelBuilders::clause(null, '', 'segmentMatch', $segmentKeys[$i + 1])) |
| 209 | + ->build() |
| 210 | + ); |
| 211 | + } |
| 212 | + $segment = $builder->build(); |
| 213 | + $segments[] = $segment; |
| 214 | + $requester->addSegment($segment); |
| 215 | + } |
| 216 | + $evaluator = new Evaluator($requester); |
| 217 | + |
| 218 | + $flag = ModelBuilders::booleanFlagWithClauses(ModelBuilders::clauseMatchingSegment($segments[0])); |
| 219 | + |
| 220 | + $result = $evaluator->evaluate($flag, $context, EvaluatorTestUtil::expectNoPrerequisiteEvals()); |
| 221 | + self::assertTrue($result->getDetail()->getValue()); |
| 222 | + } |
| 223 | + |
| 224 | + /** @dataProvider recursionDepth */ |
| 225 | + public function testSegmentCycleDetection($depth) |
| 226 | + { |
| 227 | + $context = LDContext::create('foo'); |
| 228 | + |
| 229 | + $segmentKeys = []; |
| 230 | + for ($i = 0; $i < $depth; $i++) { |
| 231 | + $segmentKeys[] = "segmentkey$i"; |
| 232 | + } |
| 233 | + $flags = []; |
| 234 | + $requester = new MockFeatureRequester(); |
| 235 | + for ($i = 0; $i < $depth; $i++) { |
| 236 | + $builder = ModelBuilders::segmentBuilder($segmentKeys[$i]); |
| 237 | + $builder->rule( |
| 238 | + ModelBuilders::segmentRuleBuilder() |
| 239 | + ->clause(ModelBuilders::clause(null, '', 'segmentMatch', $segmentKeys[($i + 1) % $depth])) |
| 240 | + ->build() |
| 241 | + ); |
| 242 | + $segment = $builder->build(); |
| 243 | + $segments[] = $segment; |
| 244 | + $requester->addSegment($segment); |
| 245 | + } |
| 246 | + $evaluator = new Evaluator($requester); |
| 247 | + |
| 248 | + $flag = ModelBuilders::booleanFlagWithClauses(ModelBuilders::clauseMatchingSegment($segments[0])); |
| 249 | + |
| 250 | + $result = $evaluator->evaluate($flag, $context, EvaluatorTestUtil::expectNoPrerequisiteEvals()); |
| 251 | + self::assertEquals(EvaluationReason::error(EvaluationReason::MALFORMED_FLAG_ERROR), $result->getDetail()->getReason()); |
| 252 | + } |
| 253 | + |
184 | 254 | private static function segmentMatchesContext(Segment $segment, LDContext $context): bool |
185 | 255 | { |
186 | 256 | $flag = ModelBuilders::booleanFlagWithClauses(ModelBuilders::clauseMatchingSegment($segment)); |
187 | 257 |
|
188 | 258 | $requester = new MockFeatureRequester(); |
189 | 259 | $requester->addSegment($segment); |
190 | | - $evaluator = new Evaluator($requester); |
| 260 | + $evaluator = new Evaluator($requester, EvaluatorTestUtil::testLogger()); |
191 | 261 |
|
192 | | - return $evaluator->evaluate($flag, $context, EvaluatorTestUtil::expectNoPrerequisiteEvals())->getDetail()->getValue(); |
| 262 | + $detail = $evaluator->evaluate($flag, $context, EvaluatorTestUtil::expectNoPrerequisiteEvals())->getDetail(); |
| 263 | + if ($detail->getValue() === null) { |
| 264 | + self::assertTrue(false, "Evaluation failed with reason: " . json_encode($detail->getReason())); |
| 265 | + } |
| 266 | + return $detail->getValue(); |
193 | 267 | } |
194 | 268 | } |
0 commit comments