mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-04 11:51:02 +01:00
Partially use ApplicationSearch in main search
Summary: Ref T4365. Primary search currently uses `PhabricatorSearchQuery` for storage, which is pretty much the same as `PhabricatorSavedQuery`, except that it's old and not used anywhere else anymore. Maniphest used to also use this table, but no longer does after Septmeber, 2013. We need to retain the class so the migration can work. This introduces `PhabricatorSearchApplicationSearchEngine` and `PhabricatorSearchDocumentQuery`, but they're both stubs that I just needed for technical reasons and/or to pass lint. The next couple patches will move logic into them and use ApplicationSearch properly. Test Plan: - Searched for stuff. - Searched for stuff with filters. - Searched for fulltext in Maniphest. - Grepped for `PhabricatorSearchQuery`. Reviewers: btrahan Reviewed By: btrahan CC: aran Maniphest Tasks: T4365 Differential Revision: https://secure.phabricator.com/D8120
This commit is contained in:
parent
e42a982445
commit
a48128d36f
12 changed files with 116 additions and 36 deletions
|
@ -1941,6 +1941,7 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorSavedQueryQuery' => 'applications/search/query/PhabricatorSavedQueryQuery.php',
|
'PhabricatorSavedQueryQuery' => 'applications/search/query/PhabricatorSavedQueryQuery.php',
|
||||||
'PhabricatorScopedEnv' => 'infrastructure/env/PhabricatorScopedEnv.php',
|
'PhabricatorScopedEnv' => 'infrastructure/env/PhabricatorScopedEnv.php',
|
||||||
'PhabricatorSearchAbstractDocument' => 'applications/search/index/PhabricatorSearchAbstractDocument.php',
|
'PhabricatorSearchAbstractDocument' => 'applications/search/index/PhabricatorSearchAbstractDocument.php',
|
||||||
|
'PhabricatorSearchApplicationSearchEngine' => 'applications/search/query/PhabricatorSearchApplicationSearchEngine.php',
|
||||||
'PhabricatorSearchAttachController' => 'applications/search/controller/PhabricatorSearchAttachController.php',
|
'PhabricatorSearchAttachController' => 'applications/search/controller/PhabricatorSearchAttachController.php',
|
||||||
'PhabricatorSearchBaseController' => 'applications/search/controller/PhabricatorSearchBaseController.php',
|
'PhabricatorSearchBaseController' => 'applications/search/controller/PhabricatorSearchBaseController.php',
|
||||||
'PhabricatorSearchConfigOptions' => 'applications/search/config/PhabricatorSearchConfigOptions.php',
|
'PhabricatorSearchConfigOptions' => 'applications/search/config/PhabricatorSearchConfigOptions.php',
|
||||||
|
@ -1950,6 +1951,7 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorSearchDocument' => 'applications/search/storage/document/PhabricatorSearchDocument.php',
|
'PhabricatorSearchDocument' => 'applications/search/storage/document/PhabricatorSearchDocument.php',
|
||||||
'PhabricatorSearchDocumentField' => 'applications/search/storage/document/PhabricatorSearchDocumentField.php',
|
'PhabricatorSearchDocumentField' => 'applications/search/storage/document/PhabricatorSearchDocumentField.php',
|
||||||
'PhabricatorSearchDocumentIndexer' => 'applications/search/index/PhabricatorSearchDocumentIndexer.php',
|
'PhabricatorSearchDocumentIndexer' => 'applications/search/index/PhabricatorSearchDocumentIndexer.php',
|
||||||
|
'PhabricatorSearchDocumentQuery' => 'applications/search/query/PhabricatorSearchDocumentQuery.php',
|
||||||
'PhabricatorSearchDocumentRelationship' => 'applications/search/storage/document/PhabricatorSearchDocumentRelationship.php',
|
'PhabricatorSearchDocumentRelationship' => 'applications/search/storage/document/PhabricatorSearchDocumentRelationship.php',
|
||||||
'PhabricatorSearchEditController' => 'applications/search/controller/PhabricatorSearchEditController.php',
|
'PhabricatorSearchEditController' => 'applications/search/controller/PhabricatorSearchEditController.php',
|
||||||
'PhabricatorSearchEngine' => 'applications/search/engine/PhabricatorSearchEngine.php',
|
'PhabricatorSearchEngine' => 'applications/search/engine/PhabricatorSearchEngine.php',
|
||||||
|
@ -4686,6 +4688,7 @@ phutil_register_library_map(array(
|
||||||
1 => 'PhabricatorPolicyInterface',
|
1 => 'PhabricatorPolicyInterface',
|
||||||
),
|
),
|
||||||
'PhabricatorSavedQueryQuery' => 'PhabricatorCursorPagedPolicyAwareQuery',
|
'PhabricatorSavedQueryQuery' => 'PhabricatorCursorPagedPolicyAwareQuery',
|
||||||
|
'PhabricatorSearchApplicationSearchEngine' => 'PhabricatorApplicationSearchEngine',
|
||||||
'PhabricatorSearchAttachController' => 'PhabricatorSearchBaseController',
|
'PhabricatorSearchAttachController' => 'PhabricatorSearchBaseController',
|
||||||
'PhabricatorSearchBaseController' => 'PhabricatorController',
|
'PhabricatorSearchBaseController' => 'PhabricatorController',
|
||||||
'PhabricatorSearchConfigOptions' => 'PhabricatorApplicationConfigOptions',
|
'PhabricatorSearchConfigOptions' => 'PhabricatorApplicationConfigOptions',
|
||||||
|
@ -4694,6 +4697,7 @@ phutil_register_library_map(array(
|
||||||
'PhabricatorSearchDeleteController' => 'PhabricatorSearchBaseController',
|
'PhabricatorSearchDeleteController' => 'PhabricatorSearchBaseController',
|
||||||
'PhabricatorSearchDocument' => 'PhabricatorSearchDAO',
|
'PhabricatorSearchDocument' => 'PhabricatorSearchDAO',
|
||||||
'PhabricatorSearchDocumentField' => 'PhabricatorSearchDAO',
|
'PhabricatorSearchDocumentField' => 'PhabricatorSearchDAO',
|
||||||
|
'PhabricatorSearchDocumentQuery' => 'PhabricatorCursorPagedPolicyAwareQuery',
|
||||||
'PhabricatorSearchDocumentRelationship' => 'PhabricatorSearchDAO',
|
'PhabricatorSearchDocumentRelationship' => 'PhabricatorSearchDAO',
|
||||||
'PhabricatorSearchEditController' => 'PhabricatorSearchBaseController',
|
'PhabricatorSearchEditController' => 'PhabricatorSearchBaseController',
|
||||||
'PhabricatorSearchEngineElastic' => 'PhabricatorSearchEngine',
|
'PhabricatorSearchEngineElastic' => 'PhabricatorSearchEngine',
|
||||||
|
|
|
@ -98,9 +98,10 @@ final class PhabricatorHomeMainController
|
||||||
|
|
||||||
return $response;
|
return $response;
|
||||||
} else if ($request->isFormPost()) {
|
} else if ($request->isFormPost()) {
|
||||||
$query = new PhabricatorSearchQuery();
|
$query = id(new PhabricatorSavedQuery())
|
||||||
$query->setQuery($jump);
|
->setEngineClassName('PhabricatorSearchApplicationSearchEngine')
|
||||||
$query->save();
|
->setParameter('query', $jump)
|
||||||
|
->save();
|
||||||
|
|
||||||
return id(new AphrontRedirectResponse())
|
return id(new AphrontRedirectResponse())
|
||||||
->setURI('/search/'.$query->getQueryKey().'/');
|
->setURI('/search/'.$query->getQueryKey().'/');
|
||||||
|
|
|
@ -406,8 +406,9 @@ final class ManiphestTaskQuery
|
||||||
|
|
||||||
// In doing a fulltext search, we first find all the PHIDs that match the
|
// In doing a fulltext search, we first find all the PHIDs that match the
|
||||||
// fulltext search, and then use that to limit the rest of the search
|
// fulltext search, and then use that to limit the rest of the search
|
||||||
$fulltext_query = new PhabricatorSearchQuery();
|
$fulltext_query = id(new PhabricatorSavedQuery())
|
||||||
$fulltext_query->setQuery($this->fullTextSearch);
|
->setEngineClassName('PhabricatorSearchApplicaionSearchEngine')
|
||||||
|
->setParameter('query', $this->fullTextSearch);
|
||||||
|
|
||||||
// NOTE: Setting this to something larger than 2^53 will raise errors in
|
// NOTE: Setting this to something larger than 2^53 will raise errors in
|
||||||
// ElasticSearch, and billions of results won't fit in memory anyway.
|
// ElasticSearch, and billions of results won't fit in memory anyway.
|
||||||
|
|
|
@ -21,14 +21,15 @@ final class PhabricatorSearchController
|
||||||
$user = $request->getUser();
|
$user = $request->getUser();
|
||||||
|
|
||||||
if ($this->key) {
|
if ($this->key) {
|
||||||
$query = id(new PhabricatorSearchQuery())->loadOneWhere(
|
$query = id(new PhabricatorSavedQuery())->loadOneWhere(
|
||||||
'queryKey = %s',
|
'queryKey = %s',
|
||||||
$this->key);
|
$this->key);
|
||||||
if (!$query) {
|
if (!$query) {
|
||||||
return new Aphront404Response();
|
return new Aphront404Response();
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
$query = new PhabricatorSearchQuery();
|
$query = id(new PhabricatorSavedQuery())
|
||||||
|
->setEngineClassName('PhabricatorSearchApplicationSearchEngine');
|
||||||
|
|
||||||
if ($request->isFormPost()) {
|
if ($request->isFormPost()) {
|
||||||
$query_str = $request->getStr('query');
|
$query_str = $request->getStr('query');
|
||||||
|
@ -45,7 +46,7 @@ final class PhabricatorSearchController
|
||||||
if ($response) {
|
if ($response) {
|
||||||
return $response;
|
return $response;
|
||||||
} else {
|
} else {
|
||||||
$query->setQuery($query_str);
|
$query->setParameter('query', $query_str);
|
||||||
|
|
||||||
if ($request->getStr('scope')) {
|
if ($request->getStr('scope')) {
|
||||||
switch ($request->getStr('scope')) {
|
switch ($request->getStr('scope')) {
|
||||||
|
@ -101,7 +102,11 @@ final class PhabricatorSearchController
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
$query->save();
|
$query->save();
|
||||||
|
} catch (AphrontQueryDuplicateKeyException $ex) {
|
||||||
|
// Someone has already executed this query.
|
||||||
|
}
|
||||||
return id(new AphrontRedirectResponse())
|
return id(new AphrontRedirectResponse())
|
||||||
->setURI('/search/'.$query->getQueryKey().'/');
|
->setURI('/search/'.$query->getQueryKey().'/');
|
||||||
}
|
}
|
||||||
|
@ -157,7 +162,7 @@ final class PhabricatorSearchController
|
||||||
id(new AphrontFormTextControl())
|
id(new AphrontFormTextControl())
|
||||||
->setLabel('Search')
|
->setLabel('Search')
|
||||||
->setName('query')
|
->setName('query')
|
||||||
->setValue($query->getQuery()))
|
->setValue($query->getParameter('query')))
|
||||||
->appendChild(
|
->appendChild(
|
||||||
id(new AphrontFormSelectControl())
|
id(new AphrontFormSelectControl())
|
||||||
->setLabel('Document Type')
|
->setLabel('Document Type')
|
||||||
|
@ -227,7 +232,7 @@ final class PhabricatorSearchController
|
||||||
if (!$request->getInt('page')) {
|
if (!$request->getInt('page')) {
|
||||||
$named = id(new PhabricatorObjectQuery())
|
$named = id(new PhabricatorObjectQuery())
|
||||||
->setViewer($user)
|
->setViewer($user)
|
||||||
->withNames(array($query->getQuery()))
|
->withNames(array($query->getParameter('queyr')))
|
||||||
->execute();
|
->execute();
|
||||||
if ($named) {
|
if ($named) {
|
||||||
$results = array_merge(array_keys($named), $results);
|
$results = array_merge(array_keys($named), $results);
|
||||||
|
|
|
@ -16,10 +16,11 @@ final class PhabricatorSearchSelectController
|
||||||
$request = $this->getRequest();
|
$request = $this->getRequest();
|
||||||
$user = $request->getUser();
|
$user = $request->getUser();
|
||||||
|
|
||||||
$query = new PhabricatorSearchQuery();
|
$query = new PhabricatorSavedQuery();
|
||||||
$query_str = $request->getStr('query');
|
$query_str = $request->getStr('query');
|
||||||
|
|
||||||
$query->setQuery($query_str);
|
$query->setEngineClassName('PhabricatorSearchApplicationSearchEngine');
|
||||||
|
$query->setParameter('query', $query_str);
|
||||||
$query->setParameter('type', $this->type);
|
$query->setParameter('type', $this->type);
|
||||||
|
|
||||||
switch ($request->getStr('filter')) {
|
switch ($request->getStr('filter')) {
|
||||||
|
|
|
@ -5,8 +5,6 @@
|
||||||
* Base class for Phabricator search engine providers. Each engine must offer
|
* Base class for Phabricator search engine providers. Each engine must offer
|
||||||
* three capabilities: indexing, searching, and reconstruction (this can be
|
* three capabilities: indexing, searching, and reconstruction (this can be
|
||||||
* stubbed out if an engine can't reasonably do it, it is used for debugging).
|
* stubbed out if an engine can't reasonably do it, it is used for debugging).
|
||||||
*
|
|
||||||
* @group search
|
|
||||||
*/
|
*/
|
||||||
abstract class PhabricatorSearchEngine {
|
abstract class PhabricatorSearchEngine {
|
||||||
|
|
||||||
|
@ -33,9 +31,9 @@ abstract class PhabricatorSearchEngine {
|
||||||
/**
|
/**
|
||||||
* Execute a search query.
|
* Execute a search query.
|
||||||
*
|
*
|
||||||
* @param PhabricatorSearchQuery A query to execute.
|
* @param PhabricatorSavedQuery A query to execute.
|
||||||
* @return list A list of matching PHIDs.
|
* @return list A list of matching PHIDs.
|
||||||
*/
|
*/
|
||||||
abstract public function executeSearch(PhabricatorSearchQuery $query);
|
abstract public function executeSearch(PhabricatorSavedQuery $query);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,8 +1,5 @@
|
||||||
<?php
|
<?php
|
||||||
|
|
||||||
/**
|
|
||||||
* @group search
|
|
||||||
*/
|
|
||||||
final class PhabricatorSearchEngineElastic extends PhabricatorSearchEngine {
|
final class PhabricatorSearchEngineElastic extends PhabricatorSearchEngine {
|
||||||
private $uri;
|
private $uri;
|
||||||
private $timeout;
|
private $timeout;
|
||||||
|
@ -94,14 +91,14 @@ final class PhabricatorSearchEngineElastic extends PhabricatorSearchEngine {
|
||||||
return $doc;
|
return $doc;
|
||||||
}
|
}
|
||||||
|
|
||||||
private function buildSpec(PhabricatorSearchQuery $query) {
|
private function buildSpec(PhabricatorSavedQuery $query) {
|
||||||
$spec = array();
|
$spec = array();
|
||||||
$filter = array();
|
$filter = array();
|
||||||
|
|
||||||
if ($query->getQuery() != '') {
|
if ($query->getParameter('query') != '') {
|
||||||
$spec[] = array(
|
$spec[] = array(
|
||||||
'field' => array(
|
'field' => array(
|
||||||
'field.corpus' => $query->getQuery(),
|
'field.corpus' => $query->getParameter('query'),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -167,7 +164,7 @@ final class PhabricatorSearchEngineElastic extends PhabricatorSearchEngine {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!$query->getQuery()) {
|
if (!$query->getParameter('query')) {
|
||||||
$spec['sort'] = array(
|
$spec['sort'] = array(
|
||||||
array('dateCreated' => 'desc'),
|
array('dateCreated' => 'desc'),
|
||||||
);
|
);
|
||||||
|
@ -179,7 +176,7 @@ final class PhabricatorSearchEngineElastic extends PhabricatorSearchEngine {
|
||||||
return $spec;
|
return $spec;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function executeSearch(PhabricatorSearchQuery $query) {
|
public function executeSearch(PhabricatorSavedQuery $query) {
|
||||||
$type = $query->getParameter('type');
|
$type = $query->getParameter('type');
|
||||||
if ($type) {
|
if ($type) {
|
||||||
$uri = "/phabricator/{$type}/_search";
|
$uri = "/phabricator/{$type}/_search";
|
||||||
|
@ -197,11 +194,13 @@ final class PhabricatorSearchEngineElastic extends PhabricatorSearchEngine {
|
||||||
// elasticsearch probably uses Lucene query syntax:
|
// elasticsearch probably uses Lucene query syntax:
|
||||||
// http://lucene.apache.org/core/3_6_1/queryparsersyntax.html
|
// http://lucene.apache.org/core/3_6_1/queryparsersyntax.html
|
||||||
// Try literal search if operator search fails.
|
// Try literal search if operator search fails.
|
||||||
if (!$query->getQuery()) {
|
if (!$query->getParameter('query')) {
|
||||||
throw $ex;
|
throw $ex;
|
||||||
}
|
}
|
||||||
$query = clone $query;
|
$query = clone $query;
|
||||||
$query->setQuery(addcslashes($query->getQuery(), '+-&|!(){}[]^"~*?:\\'));
|
$query->setQuery(
|
||||||
|
addcslashes(
|
||||||
|
$query->getParameter('query'), '+-&|!(){}[]^"~*?:\\'));
|
||||||
$response = $this->executeRequest($uri, $this->buildSpec($query));
|
$response = $this->executeRequest($uri, $this->buildSpec($query));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -142,7 +142,7 @@ final class PhabricatorSearchEngineMySQL extends PhabricatorSearchEngine {
|
||||||
return $adoc;
|
return $adoc;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function executeSearch(PhabricatorSearchQuery $query) {
|
public function executeSearch(PhabricatorSavedQuery $query) {
|
||||||
|
|
||||||
$where = array();
|
$where = array();
|
||||||
$join = array();
|
$join = array();
|
||||||
|
@ -156,7 +156,7 @@ final class PhabricatorSearchEngineMySQL extends PhabricatorSearchEngine {
|
||||||
|
|
||||||
$conn_r = $dao_doc->establishConnection('r');
|
$conn_r = $dao_doc->establishConnection('r');
|
||||||
|
|
||||||
$q = $query->getQuery();
|
$q = $query->getParameter('query');
|
||||||
|
|
||||||
if (strlen($q)) {
|
if (strlen($q)) {
|
||||||
$join[] = qsprintf(
|
$join[] = qsprintf(
|
||||||
|
@ -281,7 +281,7 @@ final class PhabricatorSearchEngineMySQL extends PhabricatorSearchEngine {
|
||||||
|
|
||||||
protected function joinRelationship(
|
protected function joinRelationship(
|
||||||
AphrontDatabaseConnection $conn,
|
AphrontDatabaseConnection $conn,
|
||||||
PhabricatorSearchQuery $query,
|
PhabricatorSavedQuery $query,
|
||||||
$field,
|
$field,
|
||||||
$type) {
|
$type) {
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,49 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
final class PhabricatorSearchApplicationSearchEngine
|
||||||
|
extends PhabricatorApplicationSearchEngine {
|
||||||
|
|
||||||
|
public function buildSavedQueryFromRequest(AphrontRequest $request) {
|
||||||
|
$saved = new PhabricatorSavedQuery();
|
||||||
|
|
||||||
|
return $saved;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function buildQueryFromSavedQuery(PhabricatorSavedQuery $saved) {
|
||||||
|
$query = id(new PhabricatorSearchDocumentQuery());
|
||||||
|
|
||||||
|
return $query;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function buildSearchForm(
|
||||||
|
AphrontFormView $form,
|
||||||
|
PhabricatorSavedQuery $saved_query) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getURI($path) {
|
||||||
|
return '/search/'.$path;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getBuiltinQueryNames() {
|
||||||
|
$names = array(
|
||||||
|
'all' => pht('All Documents'),
|
||||||
|
);
|
||||||
|
|
||||||
|
return $names;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function buildSavedQueryFromBuiltin($query_key) {
|
||||||
|
|
||||||
|
$query = $this->newSavedQuery();
|
||||||
|
$query->setQueryKey($query_key);
|
||||||
|
|
||||||
|
switch ($query_key) {
|
||||||
|
case 'all':
|
||||||
|
return $query;
|
||||||
|
}
|
||||||
|
|
||||||
|
return parent::buildSavedQueryFromBuiltin($query_key);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,14 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
final class PhabricatorSearchDocumentQuery
|
||||||
|
extends PhabricatorCursorPagedPolicyAwareQuery {
|
||||||
|
|
||||||
|
protected function loadPage() {
|
||||||
|
return array();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getQueryApplicationClass() {
|
||||||
|
return 'PhabricatorApplicationSearch';
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -1,7 +1,18 @@
|
||||||
<?php
|
<?php
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @group search
|
* Obsolete storage for saved search parameters. This class is no longer used;
|
||||||
|
* it was obsoleted by the introduction of {@class:PhabricatorSavedQuery}.
|
||||||
|
*
|
||||||
|
* This class is retained only because one of the migrations
|
||||||
|
* (`20130913.maniphest.1.migratesearch.php`) relies on it to migrate old saved
|
||||||
|
* Maniphest searches to new infrastructure. We can remove this class and the
|
||||||
|
* corresponding migration after installs have had a reasonable amount of time
|
||||||
|
* to perform it.
|
||||||
|
*
|
||||||
|
* TODO: Remove this class after 2014-09-13, roughly.
|
||||||
|
*
|
||||||
|
* @deprecated
|
||||||
*/
|
*/
|
||||||
final class PhabricatorSearchQuery extends PhabricatorSearchDAO {
|
final class PhabricatorSearchQuery extends PhabricatorSearchDAO {
|
||||||
|
|
||||||
|
|
|
@ -1,8 +1,5 @@
|
||||||
<?php
|
<?php
|
||||||
|
|
||||||
/**
|
|
||||||
* @group search
|
|
||||||
*/
|
|
||||||
final class PhabricatorSearchResultView extends AphrontView {
|
final class PhabricatorSearchResultView extends AphrontView {
|
||||||
|
|
||||||
private $handle;
|
private $handle;
|
||||||
|
@ -14,7 +11,7 @@ final class PhabricatorSearchResultView extends AphrontView {
|
||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function setQuery(PhabricatorSearchQuery $query) {
|
public function setQuery(PhabricatorSavedQuery $query) {
|
||||||
$this->query = $query;
|
$this->query = $query;
|
||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
@ -84,7 +81,7 @@ final class PhabricatorSearchResultView extends AphrontView {
|
||||||
return $str;
|
return $str;
|
||||||
}
|
}
|
||||||
|
|
||||||
$query = $this->query->getQuery();
|
$query = $this->query->getParameter('query');
|
||||||
|
|
||||||
$quoted_regexp = '/"([^"]*)"/';
|
$quoted_regexp = '/"([^"]*)"/';
|
||||||
$matches = array(1 => array());
|
$matches = array(1 => array());
|
||||||
|
|
Loading…
Reference in a new issue