Skip to content

Commit 55a716b

Browse files
committedSep 11, 2023
security #cve-2023-41336 [Autocomplete] Fix bug of not using query_builder (weaverryan)
This PR was merged into the 2.x branch.
2 parents 0950bfa + 5f0f9b5 commit 55a716b

File tree

2 files changed

+28
-14
lines changed

2 files changed

+28
-14
lines changed
 

‎src/Autocomplete/src/Form/AutocompleteEntityTypeSubscriber.php

+14-13
Original file line numberDiff line numberDiff line change
@@ -11,9 +11,7 @@
1111

1212
namespace Symfony\UX\Autocomplete\Form;
1313

14-
use Doctrine\Common\Collections\ArrayCollection;
1514
use Doctrine\ORM\EntityManagerInterface;
16-
use Doctrine\ORM\Query\Parameter;
1715
use Doctrine\ORM\Utility\PersisterHelper;
1816
use Symfony\Bridge\Doctrine\Form\Type\EntityType;
1917
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
@@ -54,13 +52,15 @@ public function preSubmit(FormEvent $event)
5452
$form = $event->getForm();
5553
$options = $form->get('autocomplete')->getConfig()->getOptions();
5654

55+
/** @var EntityManagerInterface $em */
56+
$em = $options['em'];
57+
$repository = $em->getRepository($options['class']);
58+
$queryBuilder = $options['query_builder'] ?: $repository->createQueryBuilder('o');
59+
$rootAlias = $queryBuilder->getRootAliases()[0];
60+
5761
if (!isset($data['autocomplete']) || '' === $data['autocomplete']) {
5862
$options['choices'] = [];
5963
} else {
60-
/** @var EntityManagerInterface $em */
61-
$em = $options['em'];
62-
$repository = $em->getRepository($options['class']);
63-
6464
$idField = $options['id_reader']->getIdField();
6565
$idType = PersisterHelper::getTypeOfField($idField, $em->getClassMetadata($options['class']), $em)[0];
6666

@@ -69,22 +69,23 @@ public function preSubmit(FormEvent $event)
6969
$idx = 0;
7070

7171
foreach ($data['autocomplete'] as $id) {
72-
$params[":id_$idx"] = new Parameter("id_$idx", $id, $idType);
72+
$params[":id_$idx"] = [$id, $idType];
7373
++$idx;
7474
}
7575

76-
$queryBuilder = $repository->createQueryBuilder('o');
77-
7876
if ($params) {
7977
$queryBuilder
80-
->where(sprintf("o.$idField IN (%s)", implode(', ', array_keys($params))))
81-
->setParameters(new ArrayCollection($params));
78+
->andWhere(sprintf("$rootAlias.$idField IN (%s)", implode(', ', array_keys($params))))
79+
;
80+
foreach ($params as $key => $param) {
81+
$queryBuilder->setParameter($key, $param[0], $param[1]);
82+
}
8283
}
8384

8485
$options['choices'] = $queryBuilder->getQuery()->getResult();
8586
} else {
86-
$options['choices'] = $repository->createQueryBuilder('o')
87-
->where("o.$idField = :id")
87+
$options['choices'] = $queryBuilder
88+
->andWhere("$rootAlias.$idField = :id")
8889
->setParameter('id', $data['autocomplete'], $idType)
8990
->getQuery()
9091
->getResult();

‎src/Autocomplete/tests/Functional/AutocompleteFormRenderingTest.php

+14-1
Original file line numberDiff line numberDiff line change
@@ -46,21 +46,34 @@ public function testCategoryFieldSubmitsCorrectly()
4646
$firstCat = CategoryFactory::createOne(['name' => 'First cat']);
4747
CategoryFactory::createOne(['name' => 'in space']);
4848
CategoryFactory::createOne(['name' => 'ate pizza']);
49+
$fooCat = CategoryFactory::createOne(['name' => 'I contain "foo" which CategoryAutocompleteType uses its query_builder option.']);
4950

5051
$this->browser()
5152
->throwExceptions()
5253
->get('/test-form')
5354
// the field renders empty (but the placeholder is there)
5455
->assertElementCount('#product_category_autocomplete option', 1)
5556
->assertNotContains('First cat')
57+
5658
->post('/test-form', [
5759
'body' => [
5860
'product' => ['category' => ['autocomplete' => $firstCat->getId()]],
5961
],
6062
])
63+
// the option does NOT match something returned by query_builder
64+
// so ONLY the placeholder shows up
65+
->assertElementCount('#product_category_autocomplete option', 1)
66+
->assertNotContains('First cat')
67+
68+
->assertNotContains('First cat')
69+
->post('/test-form', [
70+
'body' => [
71+
'product' => ['category' => ['autocomplete' => $fooCat->getId()]],
72+
],
73+
])
6174
// the one option + placeholder now shows up
6275
->assertElementCount('#product_category_autocomplete option', 2)
63-
->assertContains('First cat')
76+
->assertContains('which CategoryAutocompleteType uses')
6477
;
6578
}
6679

0 commit comments

Comments
 (0)
Please sign in to comment.