. */ declare(strict_types=1); namespace Fisharebest\Webtrees\Factories; use Fisharebest\Webtrees\Contracts\XrefFactoryInterface; use Fisharebest\Webtrees\Site; use Illuminate\Database\Capsule\Manager as DB; /** * Make an XREF. */ class XrefFactory implements XrefFactoryInterface { /** * Create a new XREF. * * @param string $record_type * * @return string */ public function make(string $record_type): string { return $this->generate('X', ''); } /** * @param string $prefix * @param string $suffix * * @return string */ protected function generate(string $prefix, string $suffix): string { // Lock the row, so that only one new XREF may be generated at a time. $num = (int) DB::table('site_setting') ->where('setting_name', '=', 'next_xref') ->lockForUpdate() ->value('setting_value'); $increment = 1.0; do { $num += (int) $increment; // This exponential increment allows us to scan over large blocks of // existing data in a reasonable time. $increment *= 1.01; $xref = $prefix . $num . $suffix; // Records may already exist with this sequence number. $already_used = DB::table('individuals')->where('i_id', '=', $xref)->exists() || DB::table('families')->where('f_id', '=', $xref)->exists() || DB::table('sources')->where('s_id', '=', $xref)->exists() || DB::table('media')->where('m_id', '=', $xref)->exists() || DB::table('other')->where('o_id', '=', $xref)->exists() || DB::table('change')->where('xref', '=', $xref)->exists(); } while ($already_used); Site::setPreference('next_xref', (string) $num); return $xref; } }