. */ declare(strict_types=1); namespace Fisharebest\Webtrees; use Closure; use Fisharebest\Webtrees\Contracts\UserInterface; use Fisharebest\Webtrees\Services\PendingChangesService; use InvalidArgumentException; use League\Flysystem\FilesystemOperator; use function array_key_exists; use function date; use function is_string; use function str_starts_with; use function strtoupper; use function substr_replace; use function trigger_error; /** * Provide an interface to the wt_gedcom table. */ class Tree { private const array RESN_PRIVACY = [ 'none' => Auth::PRIV_PRIVATE, 'privacy' => Auth::PRIV_USER, 'confidential' => Auth::PRIV_NONE, 'hidden' => Auth::PRIV_HIDE, ]; // Default values for some tree preferences. protected const array DEFAULT_PREFERENCES = [ 'CALENDAR_FORMAT' => 'gregorian', 'CHART_BOX_TAGS' => '', 'EXPAND_SOURCES' => '0', 'FAM_FACTS_QUICK' => 'ENGA,MARR,DIV', 'FORMAT_TEXT' => 'markdown', 'GEDCOM_MEDIA_PATH' => '', 'GENERATE_UIDS' => '0', 'HIDE_GEDCOM_ERRORS' => '1', 'HIDE_LIVE_PEOPLE' => '1', 'INDI_FACTS_QUICK' => 'BIRT,BURI,BAPM,CENS,DEAT,OCCU,RESI', 'KEEP_ALIVE_YEARS_BIRTH' => '', 'KEEP_ALIVE_YEARS_DEATH' => '', 'MAX_ALIVE_AGE' => '120', 'MEDIA_UPLOAD' => '1', // Auth::PRIV_USER 'META_DESCRIPTION' => '', 'META_TITLE' => Webtrees::NAME, 'NO_UPDATE_CHAN' => '0', 'PEDIGREE_ROOT_ID' => '', 'QUICK_REQUIRED_FACTS' => 'BIRT,DEAT', 'QUICK_REQUIRED_FAMFACTS' => 'MARR', 'REQUIRE_AUTHENTICATION' => '0', 'SAVE_WATERMARK_IMAGE' => '0', 'SHOW_AGE_DIFF' => '0', 'SHOW_COUNTER' => '1', 'SHOW_DEAD_PEOPLE' => '2', // Auth::PRIV_PRIVATE 'SHOW_EST_LIST_DATES' => '0', 'SHOW_FACT_ICONS' => '1', 'SHOW_GEDCOM_RECORD' => '0', 'SHOW_HIGHLIGHT_IMAGES' => '1', 'SHOW_LEVEL2_NOTES' => '1', 'SHOW_LIVING_NAMES' => '1', // Auth::PRIV_USER 'SHOW_MEDIA_DOWNLOAD' => '0', 'SHOW_NO_WATERMARK' => '1', // Auth::PRIV_USER 'SHOW_PARENTS_AGE' => '1', 'SHOW_PEDIGREE_PLACES' => '9', 'SHOW_PEDIGREE_PLACES_SUFFIX' => '0', 'SHOW_PRIVATE_RELATIONSHIPS' => '1', 'SHOW_RELATIVES_EVENTS' => '_BIRT_CHIL,_BIRT_SIBL,_MARR_CHIL,_MARR_PARE,_DEAT_CHIL,_DEAT_PARE,_DEAT_GPAR,_DEAT_SIBL,_DEAT_SPOU', 'SUBLIST_TRIGGER_I' => '200', 'SURNAME_LIST_STYLE' => 'style2', 'SURNAME_TRADITION' => 'paternal', 'USE_SILHOUETTE' => '1', 'WORD_WRAPPED_NOTES' => '0', ]; private bool $default_resn_loaded = false; /** @var array Default access rules for facts in this tree */ private array $fact_privacy = []; /** @var array Default access rules for individuals in this tree */ private array $individual_privacy = []; /** @var array> Default access rules for individual facts in this tree */ private array $individual_fact_privacy = []; /** @var array Cached copy of the wt_gedcom_setting table. */ private array $preferences = []; /** @var array> Cached copy of the wt_user_gedcom_setting table. */ private array $user_preferences = []; public function __construct( private readonly int $id, private readonly string $name, private readonly string $title, private readonly string $media_folder, private readonly string $gedcom_filename, private readonly bool $imported, private readonly bool $private, private readonly int|null $contact_user_id, private readonly int|null $support_user_id, ) { } /** * @param object{ * gedcom_id:string|int, * gedcom_name:string, * title:string, * media_folder:string, * gedcom_filename:string, * private:string|int, * imported:string|int, * contact_user_id:string|int|null, * support_user_id:string|int|null, * } $row */ public static function fromDB(object $row): self { return new self( id: (int) $row->gedcom_id, name: $row->gedcom_name, title: $row->title, media_folder: $row->media_folder, gedcom_filename: $row->gedcom_filename, imported: (bool) $row->imported, private: (bool) $row->private, contact_user_id: $row->contact_user_id === null ? null : (int) $row->contact_user_id, support_user_id: $row->support_user_id === null ? null : (int) $row->support_user_id, ); } public function setPreference(string $setting_name, string $setting_value): self { switch ($setting_name) { case 'CONTACT_USER_ID': case 'WEBMASTER_USER_ID': trigger_error('Deprecated since 2.2.6 - update the table directly'); DB::table('gedcom') ->where('gedcom_id', '=', $this->id) ->update(['support_user_id' => $setting_value === '' ? null : (int) $setting_value]); return $this; case 'imported': case 'REQUIRE_AUTHENTICATION': trigger_error('Deprecated since 2.2.6 - update the table directly'); DB::table('gedcom') ->where('gedcom_id', '=', $this->id) ->update(['private' => (int) $setting_value]); return $this; case 'gedcom_filename': case 'MEDIA_DIRECTORY': case 'title': trigger_error('Deprecated since 2.2.6 - update the table directly'); DB::table('gedcom') ->where('gedcom_id', '=', $this->id) ->update(['title' => $setting_value]); return $this; } if ($setting_value !== $this->getPreference($setting_name)) { DB::table('gedcom_setting')->updateOrInsert([ 'gedcom_id' => $this->id, 'setting_name' => $setting_name, ], [ 'setting_value' => $setting_value, ]); $this->preferences[$setting_name] = $setting_value; Log::addConfigurationLog('Tree preference "' . $setting_name . '" set to "' . $setting_value . '"', $this); } return $this; } public function getPreference(string $setting_name, string|null $default = null): string { switch ($setting_name) { case 'CONTACT_USER_ID': trigger_error('Deprecated since 2.2.6 - use Tree::contactUserId().', E_USER_DEPRECATED); return (string) $this->contactUserId(); case 'gedcom_filename': trigger_error('Deprecated since 2.2.6 - use Tree::gedcomFile().', E_USER_DEPRECATED); return $this->gedcomFilename(); case 'imported': trigger_error('Deprecated since 2.2.6 - use Tree::imported().', E_USER_DEPRECATED); return $this->imported() ? '1' : ''; case 'MEDIA_DIRECTORY': trigger_error('Deprecated since 2.2.6 - use Tree::mediaFolder().', E_USER_DEPRECATED); return $this->mediaFolder(); case 'REQUIRE_AUTHENTICATION': trigger_error('Deprecated since 2.2.6 - use Tree::private().', E_USER_DEPRECATED); return $this->private() ? '1' : ''; case 'WEBMASTER_USER_ID': trigger_error('Deprecated since 2.2.6 - use Tree::supportUserId().', E_USER_DEPRECATED); return (string) $this->supportUserId(); case 'title': trigger_error('Deprecated since 2.2.6 - use Tree::title().', E_USER_DEPRECATED); return $this->title(); } if ($this->preferences === []) { $this->preferences = DB::table('gedcom_setting') ->where('gedcom_id', '=', $this->id) ->pluck('setting_value', 'setting_name') ->all(); } return $this->preferences[$setting_name] ?? $default ?? self::DEFAULT_PREFERENCES[$setting_name] ?? ''; } public function gedcomFilename(): string { return $this->gedcom_filename; } public function imported(): bool { return $this->imported; } public function mediaFolder(): string { return $this->media_folder; } public function name(): string { return $this->name; } public function private(): bool { return $this->private; } public function title(): string { return $this->title; } public function contactUserId(): int|null { return $this->contact_user_id; } public function supportUserId(): int|null { return $this->support_user_id; } /** * The fact-level privacy for this tree. * * @return array */ public function getFactPrivacy(): array { if (!$this->default_resn_loaded) { $this->loadDefaultResn(); } return $this->fact_privacy; } /** * The individual-level privacy for this tree. * * @return array */ public function getIndividualPrivacy(): array { if (!$this->default_resn_loaded) { $this->loadDefaultResn(); } return $this->individual_privacy; } /** * The individual-fact-level privacy for this tree. * * @return array> */ public function getIndividualFactPrivacy(): array { if (!$this->default_resn_loaded) { $this->loadDefaultResn(); } return $this->individual_fact_privacy; } private function loadDefaultResn(): void { $rows = DB::table('default_resn') ->where('gedcom_id', '=', $this->id) ->get(); foreach ($rows as $row) { // Convert GEDCOM privacy restriction to a webtrees access level. $row->resn = self::RESN_PRIVACY[$row->resn]; if ($row->xref !== null) { if ($row->tag_type !== null) { $this->individual_fact_privacy[$row->xref][$row->tag_type] = $row->resn; } else { $this->individual_privacy[$row->xref] = $row->resn; } } else { $this->fact_privacy[$row->tag_type] = $row->resn; } } $this->default_resn_loaded = true; } /** * Set the tree’s user-configuration settings. * * @param UserInterface $user * @param string $setting_name * @param string $setting_value * * @return self */ public function setUserPreference(UserInterface $user, string $setting_name, string $setting_value): self { if ($this->getUserPreference($user, $setting_name) !== $setting_value) { // Update the database DB::table('user_gedcom_setting')->updateOrInsert([ 'gedcom_id' => $this->id(), 'user_id' => $user->id(), 'setting_name' => $setting_name, ], [ 'setting_value' => $setting_value, ]); // Update the cache $this->user_preferences[$user->id()][$setting_name] = $setting_value; // Audit log of changes Log::addConfigurationLog('Tree preference "' . $setting_name . '" set to "' . $setting_value . '" for user "' . $user->userName() . '"', $this); } return $this; } /** * Get the tree’s user-configuration settings. * * @param UserInterface $user * @param string $setting_name * @param string $default * * @return string */ public function getUserPreference(UserInterface $user, string $setting_name, string $default = ''): string { // There are lots of settings, and we need to fetch lots of them on every page // so it is quicker to fetch them all in one go. if (!array_key_exists($user->id(), $this->user_preferences)) { $this->user_preferences[$user->id()] = DB::table('user_gedcom_setting') ->where('user_id', '=', $user->id()) ->where('gedcom_id', '=', $this->id) ->pluck('setting_value', 'setting_name') ->all(); } return $this->user_preferences[$user->id()][$setting_name] ?? $default; } /** * The ID of this tree * * @return int */ public function id(): int { return $this->id; } /** * Can a user accept changes for this tree? * * @param UserInterface $user * * @return bool */ public function canAcceptChanges(UserInterface $user): bool { return Auth::isModerator($this, $user); } public function hasPendingEdit(): bool { return DB::table('change') ->where('gedcom_id', '=', $this->id) ->where('status', '=', 'pending') ->exists(); } public function createRecord(string $gedcom): GedcomRecord { if (preg_match('/^0 @@ ([_A-Z]+)/', $gedcom, $match) !== 1) { throw new InvalidArgumentException('GedcomRecord::createRecord(' . $gedcom . ') does not begin 0 @@'); } $xref = Registry::xrefFactory()->make($match[1]); $gedcom = substr_replace($gedcom, $xref, 3, 0); // Create a change record $today = strtoupper(date('d M Y')); $now = date('H:i:s'); $gedcom .= "\n1 CHAN\n2 DATE " . $today . "\n3 TIME " . $now . "\n2 _WT_USER " . Auth::user()->userName(); // Create a pending change DB::table('change')->insert([ 'gedcom_id' => $this->id, 'xref' => $xref, 'old_gedcom' => '', 'new_gedcom' => $gedcom, 'status' => 'pending', 'user_id' => Auth::id(), ]); // Accept this pending change if (Auth::user()->getPreference(UserInterface::PREF_AUTO_ACCEPT_EDITS) === '1') { $record = Registry::gedcomRecordFactory()->new($xref, $gedcom, null, $this); $pending_changes_service = Registry::container()->get(PendingChangesService::class); $pending_changes_service->acceptRecord($record); return $record; } return Registry::gedcomRecordFactory()->new($xref, '', $gedcom, $this); } public function createFamily(string $gedcom): GedcomRecord { if (!str_starts_with($gedcom, '0 @@ FAM')) { throw new InvalidArgumentException('GedcomRecord::createFamily(' . $gedcom . ') does not begin 0 @@ FAM'); } $xref = Registry::xrefFactory()->make(Family::RECORD_TYPE); $gedcom = substr_replace($gedcom, $xref, 3, 0); // Create a change record $today = strtoupper(date('d M Y')); $now = date('H:i:s'); $gedcom .= "\n1 CHAN\n2 DATE " . $today . "\n3 TIME " . $now . "\n2 _WT_USER " . Auth::user()->userName(); // Create a pending change DB::table('change')->insert([ 'gedcom_id' => $this->id, 'xref' => $xref, 'old_gedcom' => '', 'new_gedcom' => $gedcom, 'status' => 'pending', 'user_id' => Auth::id(), ]); // Accept this pending change if (Auth::user()->getPreference(UserInterface::PREF_AUTO_ACCEPT_EDITS) === '1') { $record = Registry::familyFactory()->new($xref, $gedcom, null, $this); $pending_changes_service = Registry::container()->get(PendingChangesService::class); $pending_changes_service->acceptRecord($record); return $record; } return Registry::familyFactory()->new($xref, '', $gedcom, $this); } public function createIndividual(string $gedcom): Individual { if (!str_starts_with($gedcom, '0 @@ INDI')) { throw new InvalidArgumentException('GedcomRecord::createIndividual(' . $gedcom . ') does not begin 0 @@ INDI'); } $xref = Registry::xrefFactory()->make(Individual::RECORD_TYPE); $gedcom = substr_replace($gedcom, $xref, 3, 0); // Create a change record $today = strtoupper(date('d M Y')); $now = date('H:i:s'); $gedcom .= "\n1 CHAN\n2 DATE " . $today . "\n3 TIME " . $now . "\n2 _WT_USER " . Auth::user()->userName(); // Create a pending change DB::table('change')->insert([ 'gedcom_id' => $this->id, 'xref' => $xref, 'old_gedcom' => '', 'new_gedcom' => $gedcom, 'status' => 'pending', 'user_id' => Auth::id(), ]); // Accept this pending change if (Auth::user()->getPreference(UserInterface::PREF_AUTO_ACCEPT_EDITS) === '1') { $record = Registry::individualFactory()->new($xref, $gedcom, null, $this); $pending_changes_service = Registry::container()->get(PendingChangesService::class); $pending_changes_service->acceptRecord($record); return $record; } return Registry::individualFactory()->new($xref, '', $gedcom, $this); } public function createMediaObject(string $gedcom): Media { if (!str_starts_with($gedcom, '0 @@ OBJE')) { throw new InvalidArgumentException('GedcomRecord::createIndividual(' . $gedcom . ') does not begin 0 @@ OBJE'); } $xref = Registry::xrefFactory()->make(Media::RECORD_TYPE); $gedcom = substr_replace($gedcom, $xref, 3, 0); // Create a change record $today = strtoupper(date('d M Y')); $now = date('H:i:s'); $gedcom .= "\n1 CHAN\n2 DATE " . $today . "\n3 TIME " . $now . "\n2 _WT_USER " . Auth::user()->userName(); // Create a pending change DB::table('change')->insert([ 'gedcom_id' => $this->id, 'xref' => $xref, 'old_gedcom' => '', 'new_gedcom' => $gedcom, 'status' => 'pending', 'user_id' => Auth::id(), ]); // Accept this pending change if (Auth::user()->getPreference(UserInterface::PREF_AUTO_ACCEPT_EDITS) === '1') { $record = Registry::mediaFactory()->new($xref, $gedcom, null, $this); $pending_changes_service = Registry::container()->get(PendingChangesService::class); $pending_changes_service->acceptRecord($record); return $record; } return Registry::mediaFactory()->new($xref, '', $gedcom, $this); } public function significantIndividual(UserInterface $user, string $xref = ''): Individual { if ($xref === '') { $individual = null; } else { $individual = Registry::individualFactory()->make($xref, $this); if ($individual === null) { $family = Registry::familyFactory()->make($xref, $this); if ($family instanceof Family) { $individual = $family->spouses()->first() ?? $family->children()->first(); } } } if ($individual === null && $this->getUserPreference($user, UserInterface::PREF_TREE_DEFAULT_XREF) !== '') { $individual = Registry::individualFactory()->make($this->getUserPreference($user, UserInterface::PREF_TREE_DEFAULT_XREF), $this); } if ($individual === null && $this->getUserPreference($user, UserInterface::PREF_TREE_ACCOUNT_XREF) !== '') { $individual = Registry::individualFactory()->make($this->getUserPreference($user, UserInterface::PREF_TREE_ACCOUNT_XREF), $this); } if ($individual === null && $this->getPreference('PEDIGREE_ROOT_ID') !== '') { $individual = Registry::individualFactory()->make($this->getPreference('PEDIGREE_ROOT_ID'), $this); } if ($individual === null) { $xref = DB::table('individuals') ->where('i_file', '=', $this->id()) ->min('i_id'); if (is_string($xref)) { $individual = Registry::individualFactory()->make($xref, $this); } } return $individual ?? Registry::individualFactory()->new('I', '0 @I@ INDI', null, $this); } public function mediaFilesystem(): FilesystemOperator { return Registry::filesystem()->data($this->mediaFolder()); } }