*/ public function getSongs(Playlist $playlist, ?User $user = null): Collection { throw_unless($playlist->is_smart, NonSmartPlaylistException::create($playlist)); $query = Song::query() ->typeOf(PlayableType::SONG) ->withMetaFor($user ?? $playlist->user) ->when(License::isPlus(), static fn (SongBuilder $query) => $query->accessibleBy($user)) ->when( $playlist->own_songs_only && License::isPlus(), static fn (SongBuilder $query) => $query->where('songs.owner_id', $playlist->user_id) ); $playlist->rule_groups->each(static function (RuleGroup $group, int $index) use ($query): void { $whereClosure = static function (SongBuilder $subQuery) use ($group): void { $group->rules->each(static function (Rule $rule) use ($subQuery): void { $tokens = SqlElements::fromRule($rule); $subQuery->{$tokens->clause}(...$tokens->parameters); }); }; $query->when( $index === 0, static fn (SongBuilder $query) => $query->where($whereClosure), static fn (SongBuilder $query) => $query->orWhere($whereClosure) ); }); return $query->orderBy('songs.title')->get(); } }