xref: /webtrees/app/Services/SearchService.php (revision 54c7f8df0fc7bddb0bcc1441937c30320ca68563)
132cd2800SGreg Roach<?php
232cd2800SGreg Roach/**
332cd2800SGreg Roach * webtrees: online genealogy
432cd2800SGreg Roach * Copyright (C) 2019 webtrees development team
532cd2800SGreg Roach * This program is free software: you can redistribute it and/or modify
632cd2800SGreg Roach * it under the terms of the GNU General Public License as published by
732cd2800SGreg Roach * the Free Software Foundation, either version 3 of the License, or
832cd2800SGreg Roach * (at your option) any later version.
932cd2800SGreg Roach * This program is distributed in the hope that it will be useful,
1032cd2800SGreg Roach * but WITHOUT ANY WARRANTY; without even the implied warranty of
1132cd2800SGreg Roach * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
1232cd2800SGreg Roach * GNU General Public License for more details.
1332cd2800SGreg Roach * You should have received a copy of the GNU General Public License
1432cd2800SGreg Roach * along with this program. If not, see <http://www.gnu.org/licenses/>.
1532cd2800SGreg Roach */
1632cd2800SGreg Roachdeclare(strict_types=1);
1732cd2800SGreg Roach
1832cd2800SGreg Roachnamespace Fisharebest\Webtrees\Services;
1932cd2800SGreg Roach
2032cd2800SGreg Roachuse Closure;
21dfb2cda2SGreg Roachuse Fisharebest\Webtrees\Date;
2252a8ef61SGreg Roachuse Fisharebest\Webtrees\Exceptions\InternalServerErrorException;
2332cd2800SGreg Roachuse Fisharebest\Webtrees\Family;
24a7a24840SGreg Roachuse Fisharebest\Webtrees\Gedcom;
2532cd2800SGreg Roachuse Fisharebest\Webtrees\GedcomRecord;
2652a8ef61SGreg Roachuse Fisharebest\Webtrees\I18N;
2732cd2800SGreg Roachuse Fisharebest\Webtrees\Individual;
2832cd2800SGreg Roachuse Fisharebest\Webtrees\Media;
2932cd2800SGreg Roachuse Fisharebest\Webtrees\Note;
30b68caec6SGreg Roachuse Fisharebest\Webtrees\Place;
3132cd2800SGreg Roachuse Fisharebest\Webtrees\Repository;
322d686e68SGreg Roachuse Fisharebest\Webtrees\Soundex;
3332cd2800SGreg Roachuse Fisharebest\Webtrees\Source;
3432cd2800SGreg Roachuse Fisharebest\Webtrees\Tree;
3532cd2800SGreg Roachuse Illuminate\Database\Capsule\Manager as DB;
3632cd2800SGreg Roachuse Illuminate\Database\Query\Builder;
37a7a24840SGreg Roachuse Illuminate\Database\Query\Expression;
3832cd2800SGreg Roachuse Illuminate\Database\Query\JoinClause;
3932cd2800SGreg Roachuse Illuminate\Support\Collection;
4032cd2800SGreg Roachuse stdClass;
41a7a24840SGreg Roachuse function mb_stripos;
4232cd2800SGreg Roach
4332cd2800SGreg Roach/**
4432cd2800SGreg Roach * Search trees for genealogy records.
4532cd2800SGreg Roach */
4632cd2800SGreg Roachclass SearchService
4732cd2800SGreg Roach{
48a7a24840SGreg Roach    /**
49a7a24840SGreg Roach     * @param Tree[]   $trees
50a7a24840SGreg Roach     * @param string[] $search
51a7a24840SGreg Roach     *
52*54c7f8dfSGreg Roach     * @return Collection
53*54c7f8dfSGreg Roach     * @return Family[]
54a7a24840SGreg Roach     */
55a7a24840SGreg Roach    public function searchFamilies(array $trees, array $search): Collection
56a7a24840SGreg Roach    {
57a7a24840SGreg Roach        $query = DB::table('families');
58a7a24840SGreg Roach
59a7a24840SGreg Roach        $this->whereTrees($query, 'f_file', $trees);
60a7a24840SGreg Roach        $this->whereSearch($query, 'f_gedcom', $search);
61a7a24840SGreg Roach
62a7a24840SGreg Roach        return $query
63a7a24840SGreg Roach            ->get()
6452a8ef61SGreg Roach            ->each($this->rowLimiter())
65a7a24840SGreg Roach            ->map(Family::rowMapper())
66a7a24840SGreg Roach            ->filter(GedcomRecord::accessFilter())
677f5fa3c2SGreg Roach            ->filter($this->rawGedcomFilter($search));
68a7a24840SGreg Roach    }
69a7a24840SGreg Roach
7032cd2800SGreg Roach    /**
7132cd2800SGreg Roach     * Search for families by name.
7232cd2800SGreg Roach     *
73a7a24840SGreg Roach     * @param Tree[]   $trees
74a7a24840SGreg Roach     * @param string[] $search
7532cd2800SGreg Roach     * @param int      $offset
7632cd2800SGreg Roach     * @param int      $limit
7732cd2800SGreg Roach     *
78*54c7f8dfSGreg Roach     * @return Collection
79*54c7f8dfSGreg Roach     * @return Family[]
8032cd2800SGreg Roach     */
81a7a24840SGreg Roach    public function searchFamilyNames(array $trees, array $search, int $offset = 0, int $limit = PHP_INT_MAX): Collection
8232cd2800SGreg Roach    {
8332cd2800SGreg Roach        $query = DB::table('families')
84a7a24840SGreg Roach            ->join('name AS husb_name', function (JoinClause $join): void {
8532cd2800SGreg Roach                $join
8632cd2800SGreg Roach                    ->on('husb_name.n_file', '=', 'families.f_file')
87a7a24840SGreg Roach                    ->on('husb_name.n_id', '=', 'families.f_husb');
8832cd2800SGreg Roach            })
89a7a24840SGreg Roach            ->join('name AS wife_name', function (JoinClause $join): void {
9032cd2800SGreg Roach                $join
9132cd2800SGreg Roach                    ->on('wife_name.n_file', '=', 'families.f_file')
92a7a24840SGreg Roach                    ->on('wife_name.n_id', '=', 'families.f_wife');
9332cd2800SGreg Roach            })
94a7a24840SGreg Roach            ->where('wife_name.n_type', '<>', '_MARNM')
95a7a24840SGreg Roach            ->where('husb_name.n_type', '<>', '_MARNM');
96a7a24840SGreg Roach
97a7a24840SGreg Roach        $prefix = DB::connection()->getTablePrefix();
98436c1c02SGreg Roach        $field  = DB::raw($prefix . 'husb_name.n_full || ' . $prefix . 'wife_name.n_full');
99a7a24840SGreg Roach
100a7a24840SGreg Roach        $this->whereTrees($query, 'f_file', $trees);
101a7a24840SGreg Roach        $this->whereSearch($query, $field, $search);
102a7a24840SGreg Roach
103a7a24840SGreg Roach        $query
10432cd2800SGreg Roach            ->orderBy('husb_name.n_sort')
10532cd2800SGreg Roach            ->orderBy('wife_name.n_sort')
106c0804649SGreg Roach            ->select(['families.*', 'husb_name.n_sort', 'wife_name.n_sort'])
10732cd2800SGreg Roach            ->distinct();
10832cd2800SGreg Roach
109a7a24840SGreg Roach        return $this->paginateQuery($query, Family::rowMapper(), GedcomRecord::accessFilter(), $offset, $limit);
110a7a24840SGreg Roach    }
111a7a24840SGreg Roach
112a7a24840SGreg Roach    /**
113a7a24840SGreg Roach     * @param Tree[]   $trees
114a7a24840SGreg Roach     * @param string[] $search
115a7a24840SGreg Roach     *
116*54c7f8dfSGreg Roach     * @return Collection
117*54c7f8dfSGreg Roach     * @return Individual[]
118a7a24840SGreg Roach     */
119a7a24840SGreg Roach    public function searchIndividuals(array $trees, array $search): Collection
120a7a24840SGreg Roach    {
121a7a24840SGreg Roach        $query = DB::table('individuals');
122a7a24840SGreg Roach
123a7a24840SGreg Roach        $this->whereTrees($query, 'i_file', $trees);
124a7a24840SGreg Roach        $this->whereSearch($query, 'i_gedcom', $search);
125a7a24840SGreg Roach
126a7a24840SGreg Roach        return $query
127a7a24840SGreg Roach            ->get()
12852a8ef61SGreg Roach            ->each($this->rowLimiter())
129a7a24840SGreg Roach            ->map(Individual::rowMapper())
130a7a24840SGreg Roach            ->filter(GedcomRecord::accessFilter())
1317f5fa3c2SGreg Roach            ->filter($this->rawGedcomFilter($search));
13232cd2800SGreg Roach    }
13332cd2800SGreg Roach
13432cd2800SGreg Roach    /**
13532cd2800SGreg Roach     * Search for individuals by name.
13632cd2800SGreg Roach     *
137a7a24840SGreg Roach     * @param Tree[]   $trees
138a7a24840SGreg Roach     * @param string[] $search
13932cd2800SGreg Roach     * @param int      $offset
14032cd2800SGreg Roach     * @param int      $limit
14132cd2800SGreg Roach     *
142*54c7f8dfSGreg Roach     * @return Collection
143*54c7f8dfSGreg Roach     * @return Individual[]
14432cd2800SGreg Roach     */
145a7a24840SGreg Roach    public function searchIndividualNames(array $trees, array $search, int $offset = 0, int $limit = PHP_INT_MAX): Collection
14632cd2800SGreg Roach    {
14732cd2800SGreg Roach        $query = DB::table('individuals')
148a7a24840SGreg Roach            ->join('name', function (JoinClause $join): void {
14932cd2800SGreg Roach                $join
15032cd2800SGreg Roach                    ->on('name.n_file', '=', 'individuals.i_file')
151a7a24840SGreg Roach                    ->on('name.n_id', '=', 'individuals.i_id');
15232cd2800SGreg Roach            })
153e84cf2deSGreg Roach            ->orderBy('n_sort')
154c0804649SGreg Roach            ->select(['individuals.*', 'n_sort', 'n_num'])
15532cd2800SGreg Roach            ->distinct();
15632cd2800SGreg Roach
157a7a24840SGreg Roach        $this->whereTrees($query, 'i_file', $trees);
158a7a24840SGreg Roach        $this->whereSearch($query, 'n_full', $search);
159a7a24840SGreg Roach
160a7a24840SGreg Roach        return $this->paginateQuery($query, Individual::rowMapper(), GedcomRecord::accessFilter(), $offset, $limit);
16132cd2800SGreg Roach    }
16232cd2800SGreg Roach
16332cd2800SGreg Roach    /**
16432cd2800SGreg Roach     * Search for media objects.
16532cd2800SGreg Roach     *
166a7a24840SGreg Roach     * @param Tree[]   $trees
167a7a24840SGreg Roach     * @param string[] $search
16832cd2800SGreg Roach     * @param int      $offset
16932cd2800SGreg Roach     * @param int      $limit
17032cd2800SGreg Roach     *
171*54c7f8dfSGreg Roach     * @return Collection
172*54c7f8dfSGreg Roach     * @return Media[]
17332cd2800SGreg Roach     */
174a7a24840SGreg Roach    public function searchMedia(array $trees, array $search, int $offset = 0, int $limit = PHP_INT_MAX): Collection
17532cd2800SGreg Roach    {
176a7a24840SGreg Roach        $query = DB::table('media');
17732cd2800SGreg Roach
178a7a24840SGreg Roach        $this->whereTrees($query, 'media.m_file', $trees);
179a7a24840SGreg Roach        $this->whereSearch($query, 'm_gedcom', $search);
180a7a24840SGreg Roach
181a7a24840SGreg Roach        return $this->paginateQuery($query, Media::rowMapper(), GedcomRecord::accessFilter(), $offset, $limit);
18232cd2800SGreg Roach    }
18332cd2800SGreg Roach
18432cd2800SGreg Roach    /**
18532cd2800SGreg Roach     * Search for notes.
18632cd2800SGreg Roach     *
187a7a24840SGreg Roach     * @param Tree[]   $trees
188a7a24840SGreg Roach     * @param string[] $search
18932cd2800SGreg Roach     * @param int      $offset
19032cd2800SGreg Roach     * @param int      $limit
19132cd2800SGreg Roach     *
192*54c7f8dfSGreg Roach     * @return Collection
193*54c7f8dfSGreg Roach     * @return Note[]
19432cd2800SGreg Roach     */
195a7a24840SGreg Roach    public function searchNotes(array $trees, array $search, int $offset = 0, int $limit = PHP_INT_MAX): Collection
19632cd2800SGreg Roach    {
19732cd2800SGreg Roach        $query = DB::table('other')
198a7a24840SGreg Roach            ->where('o_type', '=', 'NOTE');
19932cd2800SGreg Roach
200a7a24840SGreg Roach        $this->whereTrees($query, 'o_file', $trees);
201a7a24840SGreg Roach        $this->whereSearch($query, 'o_gedcom', $search);
202a7a24840SGreg Roach
203a7a24840SGreg Roach        return $this->paginateQuery($query, Note::rowMapper(), GedcomRecord::accessFilter(), $offset, $limit);
20432cd2800SGreg Roach    }
20532cd2800SGreg Roach
20632cd2800SGreg Roach    /**
20732cd2800SGreg Roach     * Search for repositories.
20832cd2800SGreg Roach     *
209a7a24840SGreg Roach     * @param Tree[]   $trees
210a7a24840SGreg Roach     * @param string[] $search
21132cd2800SGreg Roach     * @param int      $offset
21232cd2800SGreg Roach     * @param int      $limit
21332cd2800SGreg Roach     *
214*54c7f8dfSGreg Roach     * @return Collection
215*54c7f8dfSGreg Roach     * @return Repository[]
21632cd2800SGreg Roach     */
217a7a24840SGreg Roach    public function searchRepositories(array $trees, array $search, int $offset = 0, int $limit = PHP_INT_MAX): Collection
21832cd2800SGreg Roach    {
21932cd2800SGreg Roach        $query = DB::table('other')
220a7a24840SGreg Roach            ->where('o_type', '=', 'REPO');
22132cd2800SGreg Roach
222a7a24840SGreg Roach        $this->whereTrees($query, 'o_file', $trees);
223a7a24840SGreg Roach        $this->whereSearch($query, 'o_gedcom', $search);
224a7a24840SGreg Roach
225a7a24840SGreg Roach        return $this->paginateQuery($query, Repository::rowMapper(), GedcomRecord::accessFilter(), $offset, $limit);
22632cd2800SGreg Roach    }
22732cd2800SGreg Roach
22832cd2800SGreg Roach    /**
229a7a24840SGreg Roach     * Search for sources.
23032cd2800SGreg Roach     *
231a7a24840SGreg Roach     * @param Tree[]   $trees
232a7a24840SGreg Roach     * @param string[] $search
23332cd2800SGreg Roach     * @param int      $offset
23432cd2800SGreg Roach     * @param int      $limit
23532cd2800SGreg Roach     *
236*54c7f8dfSGreg Roach     * @return Collection
237*54c7f8dfSGreg Roach     * @return Source[]
23832cd2800SGreg Roach     */
239a7a24840SGreg Roach    public function searchSources(array $trees, array $search, int $offset = 0, int $limit = PHP_INT_MAX): Collection
240a7a24840SGreg Roach    {
241a7a24840SGreg Roach        $query = DB::table('sources');
242a7a24840SGreg Roach
243a7a24840SGreg Roach        $this->whereTrees($query, 's_file', $trees);
244a7a24840SGreg Roach        $this->whereSearch($query, 's_gedcom', $search);
245a7a24840SGreg Roach
246a7a24840SGreg Roach        return $this->paginateQuery($query, Source::rowMapper(), GedcomRecord::accessFilter(), $offset, $limit);
247a7a24840SGreg Roach    }
248a7a24840SGreg Roach
249a7a24840SGreg Roach    /**
250a7a24840SGreg Roach     * Search for sources by name.
251a7a24840SGreg Roach     *
252a7a24840SGreg Roach     * @param Tree[]   $trees
253a7a24840SGreg Roach     * @param string[] $search
254a7a24840SGreg Roach     * @param int      $offset
255a7a24840SGreg Roach     * @param int      $limit
256a7a24840SGreg Roach     *
257*54c7f8dfSGreg Roach     * @return Collection
258*54c7f8dfSGreg Roach     * @return Source[]
259a7a24840SGreg Roach     */
260a7a24840SGreg Roach    public function searchSourcesByName(array $trees, array $search, int $offset = 0, int $limit = PHP_INT_MAX): Collection
26132cd2800SGreg Roach    {
26232cd2800SGreg Roach        $query = DB::table('sources')
263c0804649SGreg Roach            ->orderBy('s_name');
26432cd2800SGreg Roach
265a7a24840SGreg Roach        $this->whereTrees($query, 's_file', $trees);
266a7a24840SGreg Roach        $this->whereSearch($query, 's_name', $search);
267a7a24840SGreg Roach
268a7a24840SGreg Roach        return $this->paginateQuery($query, Source::rowMapper(), GedcomRecord::accessFilter(), $offset, $limit);
26932cd2800SGreg Roach    }
27032cd2800SGreg Roach
27132cd2800SGreg Roach    /**
27232cd2800SGreg Roach     * Search for submitters.
27332cd2800SGreg Roach     *
274a7a24840SGreg Roach     * @param Tree[]   $trees
275a7a24840SGreg Roach     * @param string[] $search
27632cd2800SGreg Roach     * @param int      $offset
27732cd2800SGreg Roach     * @param int      $limit
27832cd2800SGreg Roach     *
279*54c7f8dfSGreg Roach     * @return Collection
280*54c7f8dfSGreg Roach     * @return GedcomRecord[]
28132cd2800SGreg Roach     */
282a7a24840SGreg Roach    public function searchSubmitters(array $trees, array $search, int $offset = 0, int $limit = PHP_INT_MAX): Collection
28332cd2800SGreg Roach    {
28432cd2800SGreg Roach        $query = DB::table('other')
285a7a24840SGreg Roach            ->where('o_type', '=', 'SUBM');
28632cd2800SGreg Roach
287a7a24840SGreg Roach        $this->whereTrees($query, 'o_file', $trees);
288a7a24840SGreg Roach        $this->whereSearch($query, 'o_gedcom', $search);
289a7a24840SGreg Roach
290a7a24840SGreg Roach        return $this->paginateQuery($query, GedcomRecord::rowMapper(), GedcomRecord::accessFilter(), $offset, $limit);
29132cd2800SGreg Roach    }
29232cd2800SGreg Roach
29332cd2800SGreg Roach    /**
294b68caec6SGreg Roach     * Search for places.
295b68caec6SGreg Roach     *
296b68caec6SGreg Roach     * @param Tree   $tree
297b68caec6SGreg Roach     * @param string $search
298b68caec6SGreg Roach     * @param int    $offset
299b68caec6SGreg Roach     * @param int    $limit
300b68caec6SGreg Roach     *
301*54c7f8dfSGreg Roach     * @return Collection
302*54c7f8dfSGreg Roach     * @return Place[]
303b68caec6SGreg Roach     */
304b68caec6SGreg Roach    public function searchPlaces(Tree $tree, string $search, int $offset = 0, int $limit = PHP_INT_MAX): Collection
305b68caec6SGreg Roach    {
306b68caec6SGreg Roach        $query = DB::table('places AS p0')
307b68caec6SGreg Roach            ->where('p0.p_file', '=', $tree->id())
308b68caec6SGreg Roach            ->leftJoin('places AS p1', 'p1.p_id', '=', 'p0.p_parent_id')
309b68caec6SGreg Roach            ->leftJoin('places AS p2', 'p2.p_id', '=', 'p1.p_parent_id')
310b68caec6SGreg Roach            ->leftJoin('places AS p3', 'p3.p_id', '=', 'p2.p_parent_id')
311b68caec6SGreg Roach            ->leftJoin('places AS p4', 'p4.p_id', '=', 'p3.p_parent_id')
312b68caec6SGreg Roach            ->leftJoin('places AS p5', 'p5.p_id', '=', 'p4.p_parent_id')
313b68caec6SGreg Roach            ->leftJoin('places AS p6', 'p6.p_id', '=', 'p5.p_parent_id')
314b68caec6SGreg Roach            ->leftJoin('places AS p7', 'p7.p_id', '=', 'p6.p_parent_id')
315b68caec6SGreg Roach            ->leftJoin('places AS p8', 'p8.p_id', '=', 'p7.p_parent_id')
316b68caec6SGreg Roach            ->orderBy('p0.p_place')
317b68caec6SGreg Roach            ->orderBy('p1.p_place')
318b68caec6SGreg Roach            ->orderBy('p2.p_place')
319b68caec6SGreg Roach            ->orderBy('p3.p_place')
320b68caec6SGreg Roach            ->orderBy('p4.p_place')
321b68caec6SGreg Roach            ->orderBy('p5.p_place')
322b68caec6SGreg Roach            ->orderBy('p6.p_place')
323b68caec6SGreg Roach            ->orderBy('p7.p_place')
324b68caec6SGreg Roach            ->orderBy('p8.p_place')
325b68caec6SGreg Roach            ->select([
326b68caec6SGreg Roach                'p0.p_place AS place0',
327b68caec6SGreg Roach                'p1.p_place AS place1',
328b68caec6SGreg Roach                'p2.p_place AS place2',
329b68caec6SGreg Roach                'p3.p_place AS place3',
330b68caec6SGreg Roach                'p4.p_place AS place4',
331b68caec6SGreg Roach                'p5.p_place AS place5',
332b68caec6SGreg Roach                'p6.p_place AS place6',
333b68caec6SGreg Roach                'p7.p_place AS place7',
334b68caec6SGreg Roach                'p8.p_place AS place8',
335b68caec6SGreg Roach            ]);
336b68caec6SGreg Roach
337b68caec6SGreg Roach        // Filter each level of the hierarchy.
338b68caec6SGreg Roach        foreach (explode(',', $search, 9) as $level => $string) {
339b68caec6SGreg Roach            $query->whereContains('p' . $level . '.p_place', $string);
340b68caec6SGreg Roach        }
341b68caec6SGreg Roach
342b68caec6SGreg Roach        $row_mapper = function (stdClass $row) use ($tree): Place {
343b68caec6SGreg Roach            $place = implode(', ', array_filter((array) $row));
344b68caec6SGreg Roach
345b68caec6SGreg Roach            return new Place($place, $tree);
346b68caec6SGreg Roach        };
347b68caec6SGreg Roach
348a7a24840SGreg Roach        $filter = function (): bool {
349a7a24840SGreg Roach            return true;
350a7a24840SGreg Roach        };
351b68caec6SGreg Roach
352a7a24840SGreg Roach        return $this->paginateQuery($query, $row_mapper, $filter, $offset, $limit);
353a7a24840SGreg Roach    }
354b68caec6SGreg Roach
355b68caec6SGreg Roach    /**
356a5dbc5b2SGreg Roach     * @param Tree[]   $trees
357dfb2cda2SGreg Roach     * @param string[] $fields
358dfb2cda2SGreg Roach     * @param string[] $modifiers
359dfb2cda2SGreg Roach     *
360*54c7f8dfSGreg Roach     * @return Collection
361*54c7f8dfSGreg Roach     * @return Individual[]
362dfb2cda2SGreg Roach     */
363dfb2cda2SGreg Roach    public function searchIndividualsAdvanced(array $trees, array $fields, array $modifiers): Collection
364dfb2cda2SGreg Roach    {
365dfb2cda2SGreg Roach        $fields = array_filter($fields);
366dfb2cda2SGreg Roach
367dfb2cda2SGreg Roach        $query = DB::table('individuals')
368dfb2cda2SGreg Roach            ->select(['individuals.*'])
369dfb2cda2SGreg Roach            ->distinct();
370dfb2cda2SGreg Roach
371dfb2cda2SGreg Roach        $this->whereTrees($query, 'i_file', $trees);
372dfb2cda2SGreg Roach
373dfb2cda2SGreg Roach        // Join the following tables
374dfb2cda2SGreg Roach        $father_name   = false;
375dfb2cda2SGreg Roach        $mother_name   = false;
376dfb2cda2SGreg Roach        $spouse_family = false;
377dfb2cda2SGreg Roach        $indi_name     = false;
378dfb2cda2SGreg Roach        $indi_date     = false;
379dfb2cda2SGreg Roach        $fam_date      = false;
380dfb2cda2SGreg Roach        $indi_plac     = false;
381dfb2cda2SGreg Roach        $fam_plac      = false;
382dfb2cda2SGreg Roach
383dfb2cda2SGreg Roach        foreach ($fields as $field_name => $field_value) {
384dfb2cda2SGreg Roach            if ($field_value !== '') {
385dfb2cda2SGreg Roach                if (substr($field_name, 0, 14) === 'FAMC:HUSB:NAME') {
386dfb2cda2SGreg Roach                    $father_name = true;
387dfb2cda2SGreg Roach                } elseif (substr($field_name, 0, 14) === 'FAMC:WIFE:NAME') {
388dfb2cda2SGreg Roach                    $mother_name = true;
389dfb2cda2SGreg Roach                } elseif (substr($field_name, 0, 4) === 'NAME') {
390dfb2cda2SGreg Roach                    $indi_name = true;
391dfb2cda2SGreg Roach                } elseif (strpos($field_name, ':DATE') !== false) {
392dfb2cda2SGreg Roach                    if (substr($field_name, 0, 4) === 'FAMS') {
393dfb2cda2SGreg Roach                        $fam_date      = true;
394dfb2cda2SGreg Roach                        $spouse_family = true;
395dfb2cda2SGreg Roach                    } else {
396dfb2cda2SGreg Roach                        $indi_date = true;
397dfb2cda2SGreg Roach                    }
398dfb2cda2SGreg Roach                } elseif (strpos($field_name, ':PLAC') !== false) {
399dfb2cda2SGreg Roach                    if (substr($field_name, 0, 4) === 'FAMS') {
400dfb2cda2SGreg Roach                        $fam_plac      = true;
401dfb2cda2SGreg Roach                        $spouse_family = true;
402dfb2cda2SGreg Roach                    } else {
403dfb2cda2SGreg Roach                        $indi_plac = true;
404dfb2cda2SGreg Roach                    }
405dfb2cda2SGreg Roach                } elseif ($field_name === 'FAMS:NOTE') {
406dfb2cda2SGreg Roach                    $spouse_family = true;
407dfb2cda2SGreg Roach                }
408dfb2cda2SGreg Roach            }
409dfb2cda2SGreg Roach        }
410dfb2cda2SGreg Roach
411dfb2cda2SGreg Roach        if ($father_name || $mother_name) {
412dfb2cda2SGreg Roach            $query->join('link AS l1', function (JoinClause $join): void {
413dfb2cda2SGreg Roach                $join
414dfb2cda2SGreg Roach                    ->on('l1.l_file', '=', 'individuals.i_file')
415dfb2cda2SGreg Roach                    ->on('l1.l_from', '=', 'individuals.i_id')
416dfb2cda2SGreg Roach                    ->where('l1.l_type', '=', 'FAMC');
417dfb2cda2SGreg Roach            });
418dfb2cda2SGreg Roach
419dfb2cda2SGreg Roach            if ($father_name) {
420dfb2cda2SGreg Roach                $query->join('link AS l2', function (JoinClause $join): void {
421dfb2cda2SGreg Roach                    $join
422dfb2cda2SGreg Roach                        ->on('l2.l_file', '=', 'l1.l_file')
423dfb2cda2SGreg Roach                        ->on('l2.l_from', '=', 'l1.l_to')
424dfb2cda2SGreg Roach                        ->where('l2.l_type', '=', 'HUSB');
425dfb2cda2SGreg Roach                });
426dfb2cda2SGreg Roach                $query->join('name AS father_name', function (JoinClause $join): void {
427dfb2cda2SGreg Roach                    $join
428dfb2cda2SGreg Roach                        ->on('father_name.n_file', '=', 'l2.l_file')
429dfb2cda2SGreg Roach                        ->on('father_name.n_id', '=', 'l2.l_to');
430dfb2cda2SGreg Roach                });
431dfb2cda2SGreg Roach            }
432dfb2cda2SGreg Roach
433dfb2cda2SGreg Roach            if ($mother_name) {
434dfb2cda2SGreg Roach                $query->join('link AS l3', function (JoinClause $join): void {
435dfb2cda2SGreg Roach                    $join
436dfb2cda2SGreg Roach                        ->on('l3.l_file', '=', 'l1.l_file')
437dfb2cda2SGreg Roach                        ->on('l3.l_from', '=', 'l1.l_to')
438dfb2cda2SGreg Roach                        ->where('l3.l_type', '=', 'WIFE');
439dfb2cda2SGreg Roach                });
440dfb2cda2SGreg Roach                $query->join('name AS mother_name', function (JoinClause $join): void {
441dfb2cda2SGreg Roach                    $join
442dfb2cda2SGreg Roach                        ->on('mother_name.n_file', '=', 'l3.l_file')
443dfb2cda2SGreg Roach                        ->on('mother_name.n_id', '=', 'l3.l_to');
444dfb2cda2SGreg Roach                });
445dfb2cda2SGreg Roach            }
446dfb2cda2SGreg Roach        }
447dfb2cda2SGreg Roach
448dfb2cda2SGreg Roach        if ($spouse_family) {
449dfb2cda2SGreg Roach            $query->join('link AS l4', function (JoinClause $join): void {
450dfb2cda2SGreg Roach                $join
451dfb2cda2SGreg Roach                    ->on('l4.l_file', '=', 'individuals.i_file')
452dfb2cda2SGreg Roach                    ->on('l4.l_from', '=', 'individuals.i_id')
453dfb2cda2SGreg Roach                    ->where('l4.l_type', '=', 'FAMS');
454dfb2cda2SGreg Roach            });
455dfb2cda2SGreg Roach            $query->join('families AS spouse_families', function (JoinClause $join): void {
456dfb2cda2SGreg Roach                $join
457dfb2cda2SGreg Roach                    ->on('spouse_families.f_file', '=', 'l4.l_file')
458dfb2cda2SGreg Roach                    ->on('spouse_families.f_id', '=', 'l4.l_to');
459dfb2cda2SGreg Roach            });
460dfb2cda2SGreg Roach        }
461dfb2cda2SGreg Roach
462dfb2cda2SGreg Roach        if ($indi_name) {
463dfb2cda2SGreg Roach            $query->join('name AS individual_name', function (JoinClause $join): void {
464dfb2cda2SGreg Roach                $join
465dfb2cda2SGreg Roach                    ->on('individual_name.n_file', '=', 'individuals.i_file')
466dfb2cda2SGreg Roach                    ->on('individual_name.n_id', '=', 'individuals.i_id');
467dfb2cda2SGreg Roach            });
468dfb2cda2SGreg Roach        }
469dfb2cda2SGreg Roach
470dfb2cda2SGreg Roach        if ($indi_date) {
471dfb2cda2SGreg Roach            $query->join('dates AS individual_dates', function (JoinClause $join): void {
472dfb2cda2SGreg Roach                $join
473dfb2cda2SGreg Roach                    ->on('individual_dates.d_file', '=', 'individuals.i_file')
474dfb2cda2SGreg Roach                    ->on('individual_dates.d_gid', '=', 'individuals.i_id');
475dfb2cda2SGreg Roach            });
476dfb2cda2SGreg Roach        }
477dfb2cda2SGreg Roach
478dfb2cda2SGreg Roach        if ($fam_date) {
479dfb2cda2SGreg Roach            $query->join('dates AS family_dates', function (JoinClause $join): void {
480dfb2cda2SGreg Roach                $join
481dfb2cda2SGreg Roach                    ->on('family_dates.d_file', '=', 'spouse_families.f_file')
482dfb2cda2SGreg Roach                    ->on('family_dates.d_gid', '=', 'spouse_families.f_id');
483dfb2cda2SGreg Roach            });
484dfb2cda2SGreg Roach        }
485dfb2cda2SGreg Roach
486dfb2cda2SGreg Roach        if ($indi_plac) {
487dfb2cda2SGreg Roach            $query->join('placelinks AS individual_placelinks', function (JoinClause $join): void {
488dfb2cda2SGreg Roach                $join
489dfb2cda2SGreg Roach                    ->on('individual_placelinks.pl_file', '=', 'individuals.i_file')
490dfb2cda2SGreg Roach                    ->on('individual_placelinks.pl_gid', '=', 'individuals.i_id');
491dfb2cda2SGreg Roach            });
492dfb2cda2SGreg Roach            $query->join('places AS individual_places', function (JoinClause $join): void {
493dfb2cda2SGreg Roach                $join
494dfb2cda2SGreg Roach                    ->on('individual_places.p_file', '=', 'individual_placelinks.pl_file')
495dfb2cda2SGreg Roach                    ->on('individual_places.p_id', '=', 'individual_placelinks.pl_p_id');
496dfb2cda2SGreg Roach            });
497dfb2cda2SGreg Roach        }
498dfb2cda2SGreg Roach
499dfb2cda2SGreg Roach        if ($fam_plac) {
500dfb2cda2SGreg Roach            $query->join('placelinks AS familyl_placelinks', function (JoinClause $join): void {
501dfb2cda2SGreg Roach                $join
502dfb2cda2SGreg Roach                    ->on('familyl_placelinks.pl_file', '=', 'individuals.i_file')
503dfb2cda2SGreg Roach                    ->on('familyl_placelinks.pl_gid', '=', 'individuals.i_id');
504dfb2cda2SGreg Roach            });
505dfb2cda2SGreg Roach            $query->join('places AS family_places', function (JoinClause $join): void {
506dfb2cda2SGreg Roach                $join
507dfb2cda2SGreg Roach                    ->on('family_places.p_file', '=', 'familyl_placelinks.pl_file')
508dfb2cda2SGreg Roach                    ->on('family_places.p_id', '=', 'familyl_placelinks.pl_p_id');
509dfb2cda2SGreg Roach            });
510dfb2cda2SGreg Roach        }
511dfb2cda2SGreg Roach
512dfb2cda2SGreg Roach        foreach ($fields as $field_name => $field_value) {
513dfb2cda2SGreg Roach            $parts = preg_split('/:/', $field_name . '::::');
514dfb2cda2SGreg Roach            if ($parts[0] === 'NAME') {
515dfb2cda2SGreg Roach                // NAME:*
516dfb2cda2SGreg Roach                switch ($parts[1]) {
517dfb2cda2SGreg Roach                    case 'GIVN':
518dfb2cda2SGreg Roach                        switch ($modifiers[$field_name]) {
519dfb2cda2SGreg Roach                            case 'EXACT':
520dfb2cda2SGreg Roach                                $query->where('individual_name.n_givn', '=', $field_value);
521dfb2cda2SGreg Roach                                break;
522dfb2cda2SGreg Roach                            case 'BEGINS':
523dfb2cda2SGreg Roach                                $query->where('individual_name.n_givn', 'LIKE', $field_value . '%');
524dfb2cda2SGreg Roach                                break;
525dfb2cda2SGreg Roach                            case 'CONTAINS':
526dfb2cda2SGreg Roach                                $query->where('individual_name.n_givn', 'LIKE', '%' . $field_value . '%');
527dfb2cda2SGreg Roach                                break;
528dfb2cda2SGreg Roach                            case 'SDX_STD':
529dfb2cda2SGreg Roach                                $sdx = Soundex::russell($field_value);
530dfb2cda2SGreg Roach                                if ($sdx !== '') {
531dfb2cda2SGreg Roach                                    $this->wherePhonetic($query, 'individual_name.n_soundex_givn_std', $sdx);
532dfb2cda2SGreg Roach                                } else {
533dfb2cda2SGreg Roach                                    // No phonetic content? Use a substring match
534dfb2cda2SGreg Roach                                    $query->where('individual_name.n_givn', 'LIKE', '%' . $field_value . '%');
535dfb2cda2SGreg Roach                                }
536dfb2cda2SGreg Roach                                break;
537dfb2cda2SGreg Roach                            case 'SDX': // SDX uses DM by default.
538dfb2cda2SGreg Roach                            case 'SDX_DM':
539dfb2cda2SGreg Roach                                $sdx = Soundex::daitchMokotoff($field_value);
540dfb2cda2SGreg Roach                                if ($sdx !== '') {
541dfb2cda2SGreg Roach                                    $this->wherePhonetic($query, 'individual_name.n_soundex_givn_dm', $sdx);
542dfb2cda2SGreg Roach                                } else {
543dfb2cda2SGreg Roach                                    // No phonetic content? Use a substring match
544dfb2cda2SGreg Roach                                    $query->where('individual_name.n_givn', 'LIKE', '%' . $field_value . '%');
545dfb2cda2SGreg Roach                                }
546dfb2cda2SGreg Roach                                break;
547dfb2cda2SGreg Roach                        }
548dfb2cda2SGreg Roach                        break;
549dfb2cda2SGreg Roach                    case 'SURN':
550dfb2cda2SGreg Roach                        switch ($modifiers[$field_name]) {
551dfb2cda2SGreg Roach                            case 'EXACT':
552dfb2cda2SGreg Roach                                $query->where('individual_name.n_surn', '=', $field_value);
553dfb2cda2SGreg Roach                                break;
554dfb2cda2SGreg Roach                            case 'BEGINS':
555dfb2cda2SGreg Roach                                $query->where('individual_name.n_surn', 'LIKE', $field_value . '%');
556dfb2cda2SGreg Roach                                break;
557dfb2cda2SGreg Roach                            case 'CONTAINS':
558dfb2cda2SGreg Roach                                $query->where('individual_name.n_surn', 'LIKE', '%' . $field_value . '%');
559dfb2cda2SGreg Roach                                break;
560dfb2cda2SGreg Roach                            case 'SDX_STD':
561dfb2cda2SGreg Roach                                $sdx = Soundex::russell($field_value);
562dfb2cda2SGreg Roach                                if ($sdx !== '') {
563dfb2cda2SGreg Roach                                    $this->wherePhonetic($query, 'individual_name.n_soundex_surn_std', $sdx);
564dfb2cda2SGreg Roach                                } else {
565dfb2cda2SGreg Roach                                    // No phonetic content? Use a substring match
566dfb2cda2SGreg Roach                                    $query->where('individual_name.n_surn', 'LIKE', '%' . $field_value . '%');
567dfb2cda2SGreg Roach                                }
568dfb2cda2SGreg Roach                                break;
569dfb2cda2SGreg Roach                            case 'SDX': // SDX uses DM by default.
570dfb2cda2SGreg Roach                            case 'SDX_DM':
571dfb2cda2SGreg Roach                                $sdx = Soundex::daitchMokotoff($field_value);
572dfb2cda2SGreg Roach                                if ($sdx !== '') {
573dfb2cda2SGreg Roach                                    $this->wherePhonetic($query, 'individual_name.n_soundex_surn_dm', $sdx);
574dfb2cda2SGreg Roach                                } else {
575dfb2cda2SGreg Roach                                    // No phonetic content? Use a substring match
576dfb2cda2SGreg Roach                                    $query->where('individual_name.n_surn', 'LIKE', '%' . $field_value . '%');
577dfb2cda2SGreg Roach                                }
578dfb2cda2SGreg Roach                                break;
579dfb2cda2SGreg Roach                        }
580dfb2cda2SGreg Roach                        break;
581dfb2cda2SGreg Roach                    case 'NICK':
582dfb2cda2SGreg Roach                    case '_MARNM':
583dfb2cda2SGreg Roach                    case '_HEB':
584dfb2cda2SGreg Roach                    case '_AKA':
585dfb2cda2SGreg Roach                        $query
586dfb2cda2SGreg Roach                            ->where('individual_name', '=', $parts[1])
587dfb2cda2SGreg Roach                            ->where('individual_name', 'LIKE', '%' . $field_value . '%');
588dfb2cda2SGreg Roach                        break;
589dfb2cda2SGreg Roach                }
590dfb2cda2SGreg Roach                unset($fields[$field_name]);
591dfb2cda2SGreg Roach            } elseif ($parts[1] === 'DATE') {
592dfb2cda2SGreg Roach                // *:DATE
593dfb2cda2SGreg Roach                $date = new Date($field_value);
594dfb2cda2SGreg Roach                if ($date->isOK()) {
595dfb2cda2SGreg Roach                    $delta = 365 * ($modifiers[$field_name] ?? 0);
596dfb2cda2SGreg Roach                    $query
597dfb2cda2SGreg Roach                        ->where('individual_dates.d_fact', '=', $parts[0])
598dfb2cda2SGreg Roach                        ->where('individual_dates.d_julianday1', '>=', $date->minimumJulianDay() - $delta)
599dfb2cda2SGreg Roach                        ->where('individual_dates.d_julianday2', '<=', $date->minimumJulianDay() + $delta);
600dfb2cda2SGreg Roach                }
601dfb2cda2SGreg Roach                unset($fields[$field_name]);
602dfb2cda2SGreg Roach            } elseif ($parts[0] === 'FAMS' && $parts[2] === 'DATE') {
603dfb2cda2SGreg Roach                // FAMS:*:DATE
604dfb2cda2SGreg Roach                $date = new Date($field_value);
605dfb2cda2SGreg Roach                if ($date->isOK()) {
606dfb2cda2SGreg Roach                    $delta = 365 * $modifiers[$field_name];
607dfb2cda2SGreg Roach                    $query
608dfb2cda2SGreg Roach                        ->where('family_dates.d_fact', '=', $parts[1])
609dfb2cda2SGreg Roach                        ->where('family_dates.d_julianday1', '>=', $date->minimumJulianDay() - $delta)
610dfb2cda2SGreg Roach                        ->where('family_dates.d_julianday2', '<=', $date->minimumJulianDay() + $delta);
611dfb2cda2SGreg Roach                }
612dfb2cda2SGreg Roach                unset($fields[$field_name]);
613dfb2cda2SGreg Roach            } elseif ($parts[1] === 'PLAC') {
614dfb2cda2SGreg Roach                // *:PLAC
615dfb2cda2SGreg Roach                // SQL can only link a place to a person/family, not to an event.
616dfb2cda2SGreg Roach                $query->where('individual_places.p_place', 'LIKE', '%' . $field_value . '%');
617dfb2cda2SGreg Roach            } elseif ($parts[0] === 'FAMS' && $parts[2] === 'PLAC') {
618dfb2cda2SGreg Roach                // FAMS:*:PLAC
619dfb2cda2SGreg Roach                // SQL can only link a place to a person/family, not to an event.
620dfb2cda2SGreg Roach                $query->where('family_places.p_place', 'LIKE', '%' . $field_value . '%');
621dfb2cda2SGreg Roach            } elseif ($parts[0] === 'FAMC' && $parts[2] === 'NAME') {
622dfb2cda2SGreg Roach                $table = $parts[1] === 'HUSB' ? 'father_name' : 'mother_name';
623dfb2cda2SGreg Roach                // NAME:*
624dfb2cda2SGreg Roach                switch ($parts[3]) {
625dfb2cda2SGreg Roach                    case 'GIVN':
626dfb2cda2SGreg Roach                        switch ($modifiers[$field_name]) {
627dfb2cda2SGreg Roach                            case 'EXACT':
628dfb2cda2SGreg Roach                                $query->where($table . '.n_givn', '=', $field_value);
629dfb2cda2SGreg Roach                                break;
630dfb2cda2SGreg Roach                            case 'BEGINS':
631dfb2cda2SGreg Roach                                $query->where($table . '.n_givn', 'LIKE', $field_value . '%');
632dfb2cda2SGreg Roach                                break;
633dfb2cda2SGreg Roach                            case 'CONTAINS':
634dfb2cda2SGreg Roach                                $query->where($table . '.n_givn', 'LIKE', '%' . $field_value . '%');
635dfb2cda2SGreg Roach                                break;
636dfb2cda2SGreg Roach                            case 'SDX_STD':
637dfb2cda2SGreg Roach                                $sdx = Soundex::russell($field_value);
638dfb2cda2SGreg Roach                                if ($sdx !== '') {
639dfb2cda2SGreg Roach                                    $this->wherePhonetic($query, $table . '.n_soundex_givn_std', $sdx);
640dfb2cda2SGreg Roach                                } else {
641dfb2cda2SGreg Roach                                    // No phonetic content? Use a substring match
642dfb2cda2SGreg Roach                                    $query->where($table . '.n_givn', 'LIKE', '%' . $field_value . '%');
643dfb2cda2SGreg Roach                                }
644dfb2cda2SGreg Roach                                break;
645dfb2cda2SGreg Roach                            case 'SDX': // SDX uses DM by default.
646dfb2cda2SGreg Roach                            case 'SDX_DM':
647dfb2cda2SGreg Roach                                $sdx = Soundex::daitchMokotoff($field_value);
648dfb2cda2SGreg Roach                                if ($sdx !== '') {
649dfb2cda2SGreg Roach                                    $this->wherePhonetic($query, $table . '.n_soundex_givn_dm', $sdx);
650dfb2cda2SGreg Roach                                } else {
651dfb2cda2SGreg Roach                                    // No phonetic content? Use a substring match
652dfb2cda2SGreg Roach                                    $query->where($table . '.n_givn', 'LIKE', '%' . $field_value . '%');
653dfb2cda2SGreg Roach                                }
654dfb2cda2SGreg Roach                                break;
655dfb2cda2SGreg Roach                        }
656dfb2cda2SGreg Roach                        break;
657dfb2cda2SGreg Roach                    case 'SURN':
658dfb2cda2SGreg Roach                        switch ($modifiers[$field_name]) {
659dfb2cda2SGreg Roach                            case 'EXACT':
660dfb2cda2SGreg Roach                                $query->where($table . '.n_surn', '=', $field_value);
661dfb2cda2SGreg Roach                                break;
662dfb2cda2SGreg Roach                            case 'BEGINS':
663dfb2cda2SGreg Roach                                $query->where($table . '.n_surn', 'LIKE', $field_value . '%');
664dfb2cda2SGreg Roach                                break;
665dfb2cda2SGreg Roach                            case 'CONTAINS':
666dfb2cda2SGreg Roach                                $query->where($table . '.n_surn', 'LIKE', '%' . $field_value . '%');
667dfb2cda2SGreg Roach                                break;
668dfb2cda2SGreg Roach                            case 'SDX_STD':
669dfb2cda2SGreg Roach                                $sdx = Soundex::russell($field_value);
670dfb2cda2SGreg Roach                                if ($sdx !== '') {
671dfb2cda2SGreg Roach                                    $this->wherePhonetic($query, $table . '.n_soundex_surn_std', $sdx);
672dfb2cda2SGreg Roach                                } else {
673dfb2cda2SGreg Roach                                    // No phonetic content? Use a substring match
674dfb2cda2SGreg Roach                                    $query->where($table . '.n_surn', 'LIKE', '%' . $field_value . '%');
675dfb2cda2SGreg Roach                                }
676dfb2cda2SGreg Roach                                break;
677dfb2cda2SGreg Roach                            case 'SDX': // SDX uses DM by default.
678dfb2cda2SGreg Roach                            case 'SDX_DM':
679dfb2cda2SGreg Roach                                $sdx = Soundex::daitchMokotoff($field_value);
680dfb2cda2SGreg Roach                                if ($sdx !== '') {
681dfb2cda2SGreg Roach                                    $this->wherePhonetic($query, $table . '.n_soundex_surn_dm', $sdx);
682dfb2cda2SGreg Roach                                } else {
683dfb2cda2SGreg Roach                                    // No phonetic content? Use a substring match
684dfb2cda2SGreg Roach                                    $query->where($table . '.n_surn', 'LIKE', '%' . $field_value . '%');
685dfb2cda2SGreg Roach                                }
686dfb2cda2SGreg Roach                                break;
687dfb2cda2SGreg Roach                        }
688dfb2cda2SGreg Roach                        break;
689dfb2cda2SGreg Roach                }
690dfb2cda2SGreg Roach                unset($fields[$field_name]);
691dfb2cda2SGreg Roach            } elseif ($parts[0] === 'FAMS') {
692dfb2cda2SGreg Roach                // e.g. searches for occupation, religion, note, etc.
693dfb2cda2SGreg Roach                // Initial matching only.  Need PHP to apply filter.
694dfb2cda2SGreg Roach                $query->where('families.f_gedcom', 'LIKE', "%\n1 " . $parts[1] . ' %' . $field_value . '%');
695dfb2cda2SGreg Roach            } elseif ($parts[1] === 'TYPE') {
696dfb2cda2SGreg Roach                // e.g. FACT:TYPE or EVEN:TYPE
697dfb2cda2SGreg Roach                // Initial matching only.  Need PHP to apply filter.
698dfb2cda2SGreg Roach                $query->where('individuals.i_gedcom', 'LIKE', "%\n1 " . $parts[0] . '%\n2 TYPE %' . $field_value . '%');
699dfb2cda2SGreg Roach            } else {
700dfb2cda2SGreg Roach                // e.g. searches for occupation, religion, note, etc.
701dfb2cda2SGreg Roach                // Initial matching only.  Need PHP to apply filter.
702dfb2cda2SGreg Roach                $query->where('individuals.i_gedcom', 'LIKE', "%\n1 " . $parts[0] . ' %' . $field_value . '%');
703dfb2cda2SGreg Roach            }
704dfb2cda2SGreg Roach        }
705dfb2cda2SGreg Roach
706dfb2cda2SGreg Roach        return $query
707dfb2cda2SGreg Roach            ->get()
70852a8ef61SGreg Roach            ->each($this->rowLimiter())
709dfb2cda2SGreg Roach            ->map(Individual::rowMapper())
710dfb2cda2SGreg Roach            ->filter(GedcomRecord::accessFilter())
711dfb2cda2SGreg Roach            ->filter(function (Individual $individual) use ($fields): bool {
712dfb2cda2SGreg Roach                // Check for XXXX:PLAC fields, which were only partially matched by SQL
713dfb2cda2SGreg Roach                foreach ($fields as $field_name => $field_value) {
714a41e65f0SGreg Roach                    $regex = '/' . preg_quote($field_value, '/') . '/i';
715dfb2cda2SGreg Roach
716dfb2cda2SGreg Roach                    $parts = preg_split('/:/', $field_name . '::::');
717dfb2cda2SGreg Roach
718dfb2cda2SGreg Roach                    if ($parts[1] === 'PLAC') {
719dfb2cda2SGreg Roach                        // *:PLAC
720a41e65f0SGreg Roach                        foreach ($individual->facts([$parts[0]]) as $fact) {
721392561bbSGreg Roach                            if (preg_match($regex, $fact->place()->gedcomName())) {
722a41e65f0SGreg Roach                                return true;
723dfb2cda2SGreg Roach                            }
724a41e65f0SGreg Roach                        }
725dfb2cda2SGreg Roach                    } elseif ($parts[0] === 'FAMS' && $parts[2] === 'PLAC') {
726dfb2cda2SGreg Roach                        // FAMS:*:PLAC
72739ca88baSGreg Roach                        foreach ($individual->spouseFamilies() as $family) {
728a41e65f0SGreg Roach                            foreach ($family->facts([$parts[1]]) as $fact) {
729392561bbSGreg Roach                                if (preg_match($regex, $fact->place()->gedcomName())) {
730a41e65f0SGreg Roach                                    return true;
731a41e65f0SGreg Roach                                }
732dfb2cda2SGreg Roach                            }
733dfb2cda2SGreg Roach                        }
734dfb2cda2SGreg Roach                    } elseif ($parts[0] === 'FAMS') {
735dfb2cda2SGreg Roach                        // e.g. searches for occupation, religion, note, etc.
73639ca88baSGreg Roach                        foreach ($individual->spouseFamilies() as $family) {
737a41e65f0SGreg Roach                            foreach ($family->facts([$parts[1]]) as $fact) {
738a41e65f0SGreg Roach                                if (preg_match($regex, $fact->value())) {
739a41e65f0SGreg Roach                                    return true;
740a41e65f0SGreg Roach                                }
741dfb2cda2SGreg Roach                            }
742dfb2cda2SGreg Roach                        }
743dfb2cda2SGreg Roach                    } elseif ($parts[1] === 'TYPE') {
744dfb2cda2SGreg Roach                        // e.g. FACT:TYPE or EVEN:TYPE
745a41e65f0SGreg Roach                        foreach ($individual->facts([$parts[0]]) as $fact) {
746a41e65f0SGreg Roach                            if (preg_match($regex, $fact->attribute('TYPE'))) {
747a41e65f0SGreg Roach                                return true;
748a41e65f0SGreg Roach                            }
749dfb2cda2SGreg Roach                        }
750dfb2cda2SGreg Roach                    } else {
751dfb2cda2SGreg Roach                        // e.g. searches for occupation, religion, note, etc.
752a41e65f0SGreg Roach                        foreach ($individual->facts([$parts[0]]) as $fact) {
753a41e65f0SGreg Roach                            if (preg_match($regex, $fact->value())) {
754a41e65f0SGreg Roach                                return true;
755a41e65f0SGreg Roach                            }
756dfb2cda2SGreg Roach                        }
757dfb2cda2SGreg Roach                    }
758dfb2cda2SGreg Roach
759a41e65f0SGreg Roach                    // No match
760dfb2cda2SGreg Roach                    return false;
761dfb2cda2SGreg Roach                }
762dfb2cda2SGreg Roach
763dfb2cda2SGreg Roach                return true;
764dfb2cda2SGreg Roach            });
765dfb2cda2SGreg Roach    }
766dfb2cda2SGreg Roach
767dfb2cda2SGreg Roach    /**
768dfb2cda2SGreg Roach     * @param string $soundex
769dfb2cda2SGreg Roach     * @param string $lastname
770dfb2cda2SGreg Roach     * @param string $firstname
771dfb2cda2SGreg Roach     * @param string $place
772dfb2cda2SGreg Roach     * @param Tree[] $search_trees
773dfb2cda2SGreg Roach     *
774*54c7f8dfSGreg Roach     * @return Collection
775*54c7f8dfSGreg Roach     * @return Individual[]
776dfb2cda2SGreg Roach     */
777dfb2cda2SGreg Roach    public function searchIndividualsPhonetic(string $soundex, string $lastname, string $firstname, string $place, array $search_trees): Collection
778dfb2cda2SGreg Roach    {
779dfb2cda2SGreg Roach        switch ($soundex) {
780dfb2cda2SGreg Roach            default:
781dfb2cda2SGreg Roach            case 'Russell':
782dfb2cda2SGreg Roach                $givn_sdx   = Soundex::russell($firstname);
783dfb2cda2SGreg Roach                $surn_sdx   = Soundex::russell($lastname);
784dfb2cda2SGreg Roach                $plac_sdx   = Soundex::russell($place);
785dfb2cda2SGreg Roach                $givn_field = 'n_soundex_givn_std';
786dfb2cda2SGreg Roach                $surn_field = 'n_soundex_surn_std';
787dfb2cda2SGreg Roach                $plac_field = 'p_std_soundex';
788dfb2cda2SGreg Roach                break;
789dfb2cda2SGreg Roach            case 'DaitchM':
790dfb2cda2SGreg Roach                $givn_sdx   = Soundex::daitchMokotoff($firstname);
791dfb2cda2SGreg Roach                $surn_sdx   = Soundex::daitchMokotoff($lastname);
792dfb2cda2SGreg Roach                $plac_sdx   = Soundex::daitchMokotoff($place);
793dfb2cda2SGreg Roach                $givn_field = 'n_soundex_givn_dm';
794dfb2cda2SGreg Roach                $surn_field = 'n_soundex_surn_dm';
795dfb2cda2SGreg Roach                $plac_field = 'p_dm_soundex';
796dfb2cda2SGreg Roach                break;
797dfb2cda2SGreg Roach        }
798dfb2cda2SGreg Roach
799dfb2cda2SGreg Roach        // Nothing to search for? Return nothing.
800dfb2cda2SGreg Roach        if ($givn_sdx === '' && $surn_sdx === '' && $plac_sdx === '') {
8013fda39a7SGreg Roach            return new Collection();
802dfb2cda2SGreg Roach        }
803dfb2cda2SGreg Roach
804dfb2cda2SGreg Roach        $query = DB::table('individuals')
805dfb2cda2SGreg Roach            ->select(['individuals.*'])
806dfb2cda2SGreg Roach            ->distinct();
807dfb2cda2SGreg Roach
808dfb2cda2SGreg Roach        $this->whereTrees($query, 'i_file', $search_trees);
809dfb2cda2SGreg Roach
810dfb2cda2SGreg Roach        if ($plac_sdx !== '') {
811dfb2cda2SGreg Roach            $query->join('placelinks', function (JoinClause $join): void {
812dfb2cda2SGreg Roach                $join
813dfb2cda2SGreg Roach                    ->on('placelinks.pl_file', '=', 'individuals.i_file')
814dfb2cda2SGreg Roach                    ->on('placelinks.pl_gid', '=', 'individuals.i_id');
815dfb2cda2SGreg Roach            });
816dfb2cda2SGreg Roach            $query->join('places', function (JoinClause $join): void {
817dfb2cda2SGreg Roach                $join
818dfb2cda2SGreg Roach                    ->on('places.p_file', '=', 'placelinks.pl_file')
819dfb2cda2SGreg Roach                    ->on('places.p_id', '=', 'placelinks.pl_p_id');
820dfb2cda2SGreg Roach            });
821dfb2cda2SGreg Roach
822dfb2cda2SGreg Roach            $this->wherePhonetic($query, $plac_field, $plac_sdx);
823dfb2cda2SGreg Roach        }
824dfb2cda2SGreg Roach
825dfb2cda2SGreg Roach        if ($givn_sdx !== '' || $surn_sdx !== '') {
826dfb2cda2SGreg Roach            $query->join('name', function (JoinClause $join): void {
827dfb2cda2SGreg Roach                $join
828dfb2cda2SGreg Roach                    ->on('name.n_file', '=', 'individuals.i_file')
829dfb2cda2SGreg Roach                    ->on('name.n_id', '=', 'individuals.i_id');
830dfb2cda2SGreg Roach            });
831dfb2cda2SGreg Roach
832dfb2cda2SGreg Roach            $this->wherePhonetic($query, $givn_field, $givn_sdx);
833dfb2cda2SGreg Roach            $this->wherePhonetic($query, $surn_field, $surn_sdx);
834dfb2cda2SGreg Roach        }
835dfb2cda2SGreg Roach
836dfb2cda2SGreg Roach        return $query
837dfb2cda2SGreg Roach            ->get()
83852a8ef61SGreg Roach            ->each($this->rowLimiter())
839dfb2cda2SGreg Roach            ->map(Individual::rowMapper())
840dfb2cda2SGreg Roach            ->filter(GedcomRecord::accessFilter());
841dfb2cda2SGreg Roach    }
842dfb2cda2SGreg Roach
843dfb2cda2SGreg Roach    /**
84432cd2800SGreg Roach     * Paginate a search query.
84532cd2800SGreg Roach     *
84632cd2800SGreg Roach     * @param Builder $query      Searches the database for the desired records.
84732cd2800SGreg Roach     * @param Closure $row_mapper Converts a row from the query into a record.
848a7a24840SGreg Roach     * @param Closure $row_filter
84932cd2800SGreg Roach     * @param int     $offset     Skip this many rows.
85032cd2800SGreg Roach     * @param int     $limit      Take this many rows.
85132cd2800SGreg Roach     *
85232cd2800SGreg Roach     * @return Collection
85332cd2800SGreg Roach     */
854a7a24840SGreg Roach    private function paginateQuery(Builder $query, Closure $row_mapper, Closure $row_filter, int $offset, int $limit): Collection
85532cd2800SGreg Roach    {
85632cd2800SGreg Roach        $collection = new Collection();
85732cd2800SGreg Roach
85832cd2800SGreg Roach        foreach ($query->cursor() as $row) {
85932cd2800SGreg Roach            $record = $row_mapper($row);
860b68caec6SGreg Roach            // If the object has a method "canShow()", then use it to filter for privacy.
861a7a24840SGreg Roach            if ($row_filter($record)) {
86232cd2800SGreg Roach                if ($offset > 0) {
86332cd2800SGreg Roach                    $offset--;
86432cd2800SGreg Roach                } else {
86532cd2800SGreg Roach                    if ($limit > 0) {
86632cd2800SGreg Roach                        $collection->push($record);
86732cd2800SGreg Roach                    }
86832cd2800SGreg Roach
86932cd2800SGreg Roach                    $limit--;
87032cd2800SGreg Roach
87132cd2800SGreg Roach                    if ($limit === 0) {
87232cd2800SGreg Roach                        break;
87332cd2800SGreg Roach                    }
87432cd2800SGreg Roach                }
87532cd2800SGreg Roach            }
87632cd2800SGreg Roach        }
87732cd2800SGreg Roach
87832cd2800SGreg Roach        return $collection;
87932cd2800SGreg Roach    }
880a7a24840SGreg Roach
881a7a24840SGreg Roach    /**
882a7a24840SGreg Roach     * Apply search filters to a SQL query column.  Apply collation rules to MySQL.
883a7a24840SGreg Roach     *
884a7a24840SGreg Roach     * @param Builder           $query
885a7a24840SGreg Roach     * @param Expression|string $field
886a7a24840SGreg Roach     * @param string[]          $search_terms
887a7a24840SGreg Roach     */
888a7a24840SGreg Roach    private function whereSearch(Builder $query, $field, array $search_terms): void
889a7a24840SGreg Roach    {
890a7a24840SGreg Roach        if ($field instanceof Expression) {
891a7a24840SGreg Roach            $field = $field->getValue();
892a7a24840SGreg Roach        }
893a7a24840SGreg Roach
894a7a24840SGreg Roach        foreach ($search_terms as $search_term) {
895436c1c02SGreg Roach            $query->whereContains(DB::raw($field), $search_term);
896a7a24840SGreg Roach        }
897a7a24840SGreg Roach    }
898a7a24840SGreg Roach
899a7a24840SGreg Roach    /**
9002d686e68SGreg Roach     * Apply soundex search filters to a SQL query column.
9012d686e68SGreg Roach     *
9022d686e68SGreg Roach     * @param Builder           $query
9032d686e68SGreg Roach     * @param Expression|string $field
9042d686e68SGreg Roach     * @param string            $soundex
9052d686e68SGreg Roach     */
9062d686e68SGreg Roach    private function wherePhonetic(Builder $query, $field, string $soundex): void
9072d686e68SGreg Roach    {
9082d686e68SGreg Roach        if ($soundex !== '') {
9092d686e68SGreg Roach            $query->where(function (Builder $query) use ($soundex, $field): void {
9102d686e68SGreg Roach                foreach (explode(':', $soundex) as $sdx) {
9112d686e68SGreg Roach                    $query->orWhere($field, 'LIKE', '%' . $sdx . '%');
9122d686e68SGreg Roach                }
9132d686e68SGreg Roach            });
9142d686e68SGreg Roach        }
9152d686e68SGreg Roach    }
9162d686e68SGreg Roach
9172d686e68SGreg Roach    /**
918a7a24840SGreg Roach     * @param Builder $query
919a7a24840SGreg Roach     * @param string  $tree_id_field
920a7a24840SGreg Roach     * @param Tree[]  $trees
921a7a24840SGreg Roach     */
922a7a24840SGreg Roach    private function whereTrees(Builder $query, string $tree_id_field, array $trees): void
923a7a24840SGreg Roach    {
924a7a24840SGreg Roach        $tree_ids = array_map(function (Tree $tree) {
925a7a24840SGreg Roach            return $tree->id();
926a7a24840SGreg Roach        }, $trees);
927a7a24840SGreg Roach
928a7a24840SGreg Roach        $query->whereIn($tree_id_field, $tree_ids);
929a7a24840SGreg Roach    }
930a7a24840SGreg Roach
931a7a24840SGreg Roach    /**
932a7a24840SGreg Roach     * A closure to filter records by privacy-filtered GEDCOM data.
933a7a24840SGreg Roach     *
934a7a24840SGreg Roach     * @param array $search_terms
935a7a24840SGreg Roach     *
936a7a24840SGreg Roach     * @return Closure
937a7a24840SGreg Roach     */
938a7a24840SGreg Roach    private function rawGedcomFilter(array $search_terms): Closure
939a7a24840SGreg Roach    {
940a7a24840SGreg Roach        return function (GedcomRecord $record) use ($search_terms): bool {
941a7a24840SGreg Roach            // Ignore non-genealogy fields
942a7a24840SGreg Roach            $gedcom = preg_replace('/\n\d (?:_UID) .*/', '', $record->gedcom());
943a7a24840SGreg Roach
944a7a24840SGreg Roach            // Ignore matches in links
945a7a24840SGreg Roach            $gedcom = preg_replace('/\n\d ' . Gedcom::REGEX_TAG . '( @' . Gedcom::REGEX_XREF . '@)?/', '', $gedcom);
946a7a24840SGreg Roach
947a7a24840SGreg Roach            // Re-apply the filtering
948a7a24840SGreg Roach            foreach ($search_terms as $search_term) {
949a7a24840SGreg Roach                if (mb_stripos($gedcom, $search_term) === false) {
950a7a24840SGreg Roach                    return false;
951a7a24840SGreg Roach                }
952a7a24840SGreg Roach            }
953a7a24840SGreg Roach
954a7a24840SGreg Roach            return true;
955a7a24840SGreg Roach        };
956a7a24840SGreg Roach    }
95752a8ef61SGreg Roach
95852a8ef61SGreg Roach    /**
95952a8ef61SGreg Roach     * Searching for short or common text can give more results than the system can process.
96052a8ef61SGreg Roach     *
96152a8ef61SGreg Roach     * @param int $limit
96252a8ef61SGreg Roach     *
96352a8ef61SGreg Roach     * @return Closure
96452a8ef61SGreg Roach     */
96552a8ef61SGreg Roach    private function rowLimiter(int $limit = 1000): Closure
96652a8ef61SGreg Roach    {
96752a8ef61SGreg Roach        return function () use ($limit) {
96852a8ef61SGreg Roach            static $n = 0;
96952a8ef61SGreg Roach
97052a8ef61SGreg Roach            if (++$n > $limit) {
97152a8ef61SGreg Roach                $message = I18N::translate('The search returned too many results.');
97252a8ef61SGreg Roach
97352a8ef61SGreg Roach                throw new InternalServerErrorException($message);
97452a8ef61SGreg Roach            }
97552a8ef61SGreg Roach        };
97652a8ef61SGreg Roach    }
97732cd2800SGreg Roach}
978