xref: /webtrees/app/Factories/XrefFactory.php (revision 89412426857931ba768b5d3d9115409fe999a0dd)
1<?php
2
3/**
4 * webtrees: online genealogy
5 * Copyright (C) 2023 webtrees development team
6 * This program is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with this program. If not, see <https://www.gnu.org/licenses/>.
16 */
17
18declare(strict_types=1);
19
20namespace Fisharebest\Webtrees\Factories;
21
22use Fisharebest\Webtrees\Contracts\XrefFactoryInterface;
23use Fisharebest\Webtrees\DB;
24use Fisharebest\Webtrees\Site;
25
26/**
27 * Make an XREF.
28 */
29class XrefFactory implements XrefFactoryInterface
30{
31    /**
32     * Create a new XREF.
33     *
34     * @param string $record_type
35     *
36     * @return string
37     */
38    public function make(string $record_type): string
39    {
40        return $this->generate('X', '');
41    }
42
43    /**
44     * @param string $prefix
45     * @param string $suffix
46     *
47     * @return string
48     */
49    protected function generate(string $prefix, string $suffix): string
50    {
51        // Lock the row, so that only one new XREF may be generated at a time.
52        $num = (int) DB::table('site_setting')
53            ->where('setting_name', '=', 'next_xref')
54            ->lockForUpdate()
55            ->value('setting_value');
56
57        $increment = 1.0;
58
59        do {
60            $num += (int) $increment;
61
62            // This exponential increment allows us to scan over large blocks of
63            // existing data in a reasonable time.
64            $increment *= 1.01;
65
66            $xref = $prefix . $num . $suffix;
67
68            // Records may already exist with this sequence number.
69            $already_used =
70                DB::table('individuals')->where('i_id', '=', $xref)->exists() ||
71                DB::table('families')->where('f_id', '=', $xref)->exists() ||
72                DB::table('sources')->where('s_id', '=', $xref)->exists() ||
73                DB::table('media')->where('m_id', '=', $xref)->exists() ||
74                DB::table('other')->where('o_id', '=', $xref)->exists() ||
75                DB::table('change')->where('xref', '=', $xref)->exists();
76        } while ($already_used);
77
78        Site::setPreference('next_xref', (string) $num);
79
80        return $xref;
81    }
82}
83