]> Dogcows Code - chaz/p5-File-KDBX/blobdiff - lib/File/KDBX.pm
Remove parent Object method
[chaz/p5-File-KDBX] / lib / File / KDBX.pm
index 2e7c1e505fd7346774b0dd56fa427f7bbedab688..d02199ab6a6c5f66d71e78f6de7a9da72ea56046 100644 (file)
@@ -6,12 +6,12 @@ use strict;
 
 use Crypt::PRNG qw(random_bytes);
 use Devel::GlobalDestruction;
-use File::KDBX::Constants qw(:all);
+use File::KDBX::Constants qw(:all :icon);
 use File::KDBX::Error;
 use File::KDBX::Safe;
-use File::KDBX::Util qw(:class :coercion :empty :uuid :search erase simple_expression_query snakify);
+use File::KDBX::Util qw(:class :coercion :empty :search :uuid erase simple_expression_query snakify);
 use Hash::Util::FieldHash qw(fieldhashes);
-use List::Util qw(any);
+use List::Util qw(any first);
 use Ref::Util qw(is_ref is_arrayref is_plain_hashref);
 use Scalar::Util qw(blessed);
 use Time::Piece;
@@ -50,7 +50,7 @@ sub DESTROY { local ($., $@, $!, $^E, $?); !in_global_destruction and $_[0]->res
 
     $kdbx = $kdbx->init(%attributes);
 
-Initialize a L<File::KDBX> with a new set of attributes. Returns itself to allow method chaining.
+Initialize a L<File::KDBX> with a set of attributes. Returns itself to allow method chaining.
 
 This is called by L</new>.
 
@@ -123,9 +123,7 @@ sub STORABLE_thaw {
     # Dualvars aren't cloned as dualvars, so coerce the compression flags.
     $self->compression_flags($self->compression_flags);
 
-    for my $object (@{$self->all_groups}, @{$self->all_entries(history => 1)}) {
-        $object->kdbx($self);
-    }
+    $self->objects(history => 1)->each(sub { $_->kdbx($self) });
 }
 
 ##############################################################################
@@ -237,12 +235,12 @@ has deleted_objects => {};
 has raw             => coerce => \&to_string;
 
 # HEADERS
-has 'headers.comment'               => '', coerce => \&to_string;
-has 'headers.cipher_id'             => CIPHER_UUID_CHACHA20, coerce => \&to_uuid;
-has 'headers.compression_flags'     => COMPRESSION_GZIP, coerce => \&to_compression_constant;
-has 'headers.master_seed'           => sub { random_bytes(32) }, coerce => \&to_string;
-has 'headers.encryption_iv'         => sub { random_bytes(16) }, coerce => \&to_string;
-has 'headers.stream_start_bytes'    => sub { random_bytes(32) }, coerce => \&to_string;
+has 'headers.comment'               => '',                          coerce => \&to_string;
+has 'headers.cipher_id'             => CIPHER_UUID_CHACHA20,        coerce => \&to_uuid;
+has 'headers.compression_flags'     => COMPRESSION_GZIP,            coerce => \&to_compression_constant;
+has 'headers.master_seed'           => sub { random_bytes(32) },    coerce => \&to_string;
+has 'headers.encryption_iv'         => sub { random_bytes(16) },    coerce => \&to_string;
+has 'headers.stream_start_bytes'    => sub { random_bytes(32) },    coerce => \&to_string;
 has 'headers.kdf_parameters'        => sub {
     +{
         KDF_PARAM_UUID()        => KDF_UUID_AES,
@@ -271,14 +269,14 @@ has 'meta.master_key_changed'               => sub { gmtime },              coer
 has 'meta.master_key_change_rec'            => -1,                          coerce => \&to_number;
 has 'meta.master_key_change_force'          => -1,                          coerce => \&to_number;
 # has 'meta.memory_protection'                => {};
-has 'meta.custom_icons'                     => {};
+has 'meta.custom_icons'                     => [];
 has 'meta.recycle_bin_enabled'              => true,                        coerce => \&to_bool;
-has 'meta.recycle_bin_uuid'                 => "\0" x 16,                   coerce => \&to_uuid;
+has 'meta.recycle_bin_uuid'                 => UUID_NULL,                   coerce => \&to_uuid;
 has 'meta.recycle_bin_changed'              => sub { gmtime },              coerce => \&to_time;
-has 'meta.entry_templates_group'            => "\0" x 16,                   coerce => \&to_uuid;
+has 'meta.entry_templates_group'            => UUID_NULL,                   coerce => \&to_uuid;
 has 'meta.entry_templates_group_changed'    => sub { gmtime },              coerce => \&to_time;
-has 'meta.last_selected_group'              => "\0" x 16,                   coerce => \&to_uuid;
-has 'meta.last_top_visible_group'           => "\0" x 16,                   coerce => \&to_uuid;
+has 'meta.last_selected_group'              => UUID_NULL,                   coerce => \&to_uuid;
+has 'meta.last_top_visible_group'           => UUID_NULL,                   coerce => \&to_uuid;
 has 'meta.history_max_items'                => HISTORY_DEFAULT_MAX_ITEMS,   coerce => \&to_number;
 has 'meta.history_max_size'                 => HISTORY_DEFAULT_MAX_SIZE,    coerce => \&to_number;
 has 'meta.settings_changed'                 => sub { gmtime },              coerce => \&to_time;
@@ -361,64 +359,39 @@ sub minimum_version {
 
     return KDBX_VERSION_4_1 if any {
         nonempty $_->{name} || nonempty $_->{last_modification_time}
-    } values %{$self->custom_icons};
-
-    return KDBX_VERSION_4_1 if any {
-        nonempty $_->previous_parent_group || nonempty $_->tags ||
-        any { nonempty $_->{last_modification_time} } values %{$_->custom_data}
-    } @{$self->all_groups};
+    } @{$self->custom_icons};
+
+    return KDBX_VERSION_4_1 if $self->groups->next(sub {
+        nonempty $_->previous_parent_group ||
+        nonempty $_->tags ||
+        (any { nonempty $_->{last_modification_time} } values %{$_->custom_data})
+        # TODO replace next paragraph with this
+        # || $_->entries(history => 1)->next(sub {
+        #     nonempty $_->previous_parent_group ||
+        #     (defined $_->quality_check && !$_->quality_check) ||
+        #     (any { nonempty $_->{last_modification_time} } values %{$_->custom_data})
+        # })
+    });
 
-    return KDBX_VERSION_4_1 if any {
-        nonempty $_->previous_parent_group || (defined $_->quality_check && !$_->quality_check) ||
-        any { nonempty $_->{last_modification_time} } values %{$_->custom_data}
-    } @{$self->all_entries(history => 1)};
+    return KDBX_VERSION_4_1 if $self->entries(history => 1)->next(sub {
+        nonempty $_->previous_parent_group ||
+        (defined $_->quality_check && !$_->quality_check) ||
+        (any { nonempty $_->{last_modification_time} } values %{$_->custom_data})
+    });
 
     return KDBX_VERSION_4_0 if $self->kdf->uuid ne KDF_UUID_AES;
 
     return KDBX_VERSION_4_0 if nonempty $self->public_custom_data;
 
-    return KDBX_VERSION_4_0 if any {
+    return KDBX_VERSION_4_0 if $self->objects->next(sub {
         nonempty $_->custom_data
-    } @{$self->all_groups}, @{$self->all_entries(history => 1)};
+    });
 
     return KDBX_VERSION_3_1;
 }
 
 ##############################################################################
 
-=method add_group
-
-    $kdbx->add_group($group, %options);
-    $kdbx->add_group(%group_attributes, %options);
-
-Add a group to a database. This is equivalent to identifying a parent group and calling
-L<File::KDBX::Group/add_group> on the parent group, forwarding the arguments. Available options:
-
-=for :list
-* C<group> (aka C<parent>) - Group (object or group UUID) to add the group to (default: root group)
-
-=cut
-
-sub add_group {
-    my $self    = shift;
-    my $group   = @_ % 2 == 1 ? shift : undef;
-    my %args    = @_;
-
-    # find the right group to add the group to
-    my $parent = delete $args{group} // delete $args{parent} // $self->root;
-    ($parent) = $self->find_groups({uuid => $parent}) if !ref $parent;
-    $parent or throw 'Invalid group';
-
-    return $parent->add_group(defined $group ? $group : (), %args, kdbx => $self);
-}
-
-sub _wrap_group {
-    my $self  = shift;
-    my $group = shift;
-    require File::KDBX::Group;
-    return File::KDBX::Group->wrap($group, $self);
-}
-
 =method root
 
     $group = $kdbx->root;
@@ -428,13 +401,13 @@ Get or set a database's root group. You don't necessarily need to explicitly cre
 because it autovivifies when adding entries and groups to the database.
 
 Every database has only a single root group at a time. Some old KDB files might have multiple root groups.
-When reading such files, a single implicit root group is created to contain the other explicit groups. When
+When reading such files, a single implicit root group is created to contain the actual root groups. When
 writing to such a format, if the root group looks like it was implicitly created then it won't be written and
 the resulting file might have multiple root groups. This allows working with older files without changing
 their written internal structure while still adhering to modern semantics while the database is opened.
 
-B<WARNING:> The root group of a KDBX database contains all of the database's entries and other groups. If you
-replace the root group, you are essentially replacing the entire database contents with something else.
+The root group of a KDBX database contains all of the database's entries and other groups. If you replace the
+root group, you are essentially replacing the entire database contents with something else.
 
 =cut
 
@@ -448,6 +421,9 @@ sub root {
     return $self->_wrap_group($self->{root});
 }
 
+# Called by File::KeePass::KDBX so that a File::KDBX an be treated as a File::KDBX::Group in that both types
+# can have subgroups. File::KDBX already has a `groups' method that does something different from the
+# File::KDBX::Groups `groups' method.
 sub _kpx_groups {
     my $self = shift;
     return [] if !$self->{root};
@@ -483,35 +459,6 @@ sub _implicit_root {
     );
 }
 
-=method all_groups
-
-    \@groups = $kdbx->all_groups(%options);
-    \@groups = $kdbx->all_groups($base_group, %options);
-
-Get all groups deeply in a database, or all groups within a specified base group, in a flat array. Supported
-options:
-
-=for :list
-* C<base> - Only include groups within a base group (same as C<$base_group>) (default: root)
-* C<include_base> - Include the base group in the results (default: true)
-
-=cut
-
-sub all_groups {
-    my $self = shift;
-    my %args = @_ % 2 == 0 ? @_ : (base => shift, @_);
-    my $base = $args{base} // $self->root;
-
-    my @groups = $args{include_base} // 1 ? $self->_wrap_group($base) : ();
-
-    for my $subgroup (@{$base->{groups} || []}) {
-        my $more = $self->all_groups($subgroup);
-        push @groups, @$more;
-    }
-
-    return \@groups;
-}
-
 =method trace_lineage
 
     \@lineage = $kdbx->trace_lineage($group);
@@ -540,38 +487,160 @@ sub _trace_lineage {
     my $base = $lineage[-1] or return [];
 
     my $uuid = $object->uuid;
-    return \@lineage if any { $_->uuid eq $uuid } @{$base->groups || []}, @{$base->entries || []};
+    return \@lineage if any { $_->uuid eq $uuid } @{$base->groups}, @{$base->entries};
 
-    for my $subgroup (@{$base->groups || []}) {
+    for my $subgroup (@{$base->groups}) {
         my $result = $self->_trace_lineage($object, @lineage, $subgroup);
         return $result if $result;
     }
 }
 
-=method find_groups
+=method recycle_bin
 
-    @groups = $kdbx->find_groups($query, %options);
+    $group = $kdbx->recycle_bin;
+    $kdbx->recycle_bin($group);
 
-Find all groups deeply that match to a query. Options are the same as for L</all_groups>.
+Get or set the recycle bin group. Returns C<undef> if there is no recycle bin and L</recycle_bin_enabled> is
+false, otherwise the current recycle bin or an autovivified recycle bin group is returned.
 
-See L</QUERY> for a description of what C<$query> can be.
+=cut
+
+sub recycle_bin {
+    my $self = shift;
+    if (my $group = shift) {
+        $self->recycle_bin_uuid($group->uuid);
+        return $group;
+    }
+    my $group;
+    my $uuid = $self->recycle_bin_uuid;
+    $group = $self->groups->grep(uuid => $uuid)->next if $uuid ne UUID_NULL;
+    if (!$group && $self->recycle_bin_enabled) {
+        $group = $self->add_group(
+            name                => 'Recycle Bin',
+            icon_id             => ICON_TRASHCAN_FULL,
+            enable_auto_type    => false,
+            enable_searching    => false,
+        );
+        $self->recycle_bin_uuid($group->uuid);
+    }
+    return $group;
+}
+
+=method entry_templates
+
+    $group = $kdbx->entry_templates;
+    $kdbx->entry_templates($group);
+
+Get or set the entry templates group. May return C<undef> if unset.
 
 =cut
 
-sub find_groups {
+sub entry_templates {
     my $self = shift;
-    my $query = shift or throw 'Must provide a query';
-    my %args = @_;
-    my %all_groups = (
-        base            => $args{base},
-        include_base    => $args{include_base},
-    );
-    return @{search($self->all_groups(%all_groups), is_arrayref($query) ? @$query : $query)};
+    if (my $group = shift) {
+        $self->entry_templates_group($group->uuid);
+        return $group;
+    }
+    my $uuid = $self->entry_templates_group;
+    return if $uuid eq UUID_NULL;
+    return $self->groups->grep(uuid => $uuid)->next;
 }
 
-sub remove {
+=method last_selected
+
+    $group = $kdbx->last_selected;
+    $kdbx->last_selected($group);
+
+Get or set the last selected group. May return C<undef> if unset.
+
+=cut
+
+sub last_selected {
     my $self = shift;
-    my $object = shift;
+    if (my $group = shift) {
+        $self->last_selected_group($group->uuid);
+        return $group;
+    }
+    my $uuid = $self->last_selected_group;
+    return if $uuid eq UUID_NULL;
+    return $self->groups->grep(uuid => $uuid)->next;
+}
+
+=method last_top_visible
+
+    $group = $kdbx->last_top_visible;
+    $kdbx->last_top_visible($group);
+
+Get or set the last top visible group. May return C<undef> if unset.
+
+=cut
+
+sub last_top_visible {
+    my $self = shift;
+    if (my $group = shift) {
+        $self->last_top_visible_group($group->uuid);
+        return $group;
+    }
+    my $uuid = $self->last_top_visible_group;
+    return if $uuid eq UUID_NULL;
+    return $self->groups->grep(uuid => $uuid)->next;
+}
+
+##############################################################################
+
+=method add_group
+
+    $kdbx->add_group($group);
+    $kdbx->add_group(%group_attributes, %options);
+
+Add a group to a database. This is equivalent to identifying a parent group and calling
+L<File::KDBX::Group/add_group> on the parent group, forwarding the arguments. Available options:
+
+=for :list
+* C<group> (aka C<parent>) - Group object or group UUID to add the group to (default: root group)
+
+=cut
+
+sub add_group {
+    my $self    = shift;
+    my $group   = @_ % 2 == 1 ? shift : undef;
+    my %args    = @_;
+
+    # find the right group to add the group to
+    my $parent = delete $args{group} // delete $args{parent} // $self->root;
+    $parent = $self->groups->grep({uuid => $parent})->next if !ref $parent;
+    $parent or throw 'Invalid group';
+
+    return $parent->add_group(defined $group ? $group : (), %args, kdbx => $self);
+}
+
+sub _wrap_group {
+    my $self  = shift;
+    my $group = shift;
+    require File::KDBX::Group;
+    return File::KDBX::Group->wrap($group, $self);
+}
+
+=method groups
+
+    \&iterator = $kdbx->groups(%options);
+    \&iterator = $kdbx->groups($base_group, %options);
+
+Get an iterator over I<groups> within a database. Options:
+
+=for :list
+* C<base> - Only include groups within a base group (same as C<$base_group>) (default: L</root>)
+* C<inclusive> - Include the base group in the results (default: true)
+* C<algorithm> - Search algorithm, one of C<ids>, C<bfs> or C<dfs> (default: C<ids>)
+
+=cut
+
+sub groups {
+    my $self = shift;
+    my %args = @_ % 2 == 0 ? @_ : (base => shift, @_);
+    my $base = delete $args{base} // $self->root;
+
+    return $base->groups_deeply(%args);
 }
 
 ##############################################################################
@@ -585,7 +654,7 @@ Add a entry to a database. This is equivalent to identifying a parent group and
 L<File::KDBX::Group/add_entry> on the parent group, forwarding the arguments. Available options:
 
 =for :list
-* C<group> (aka C<parent>) - Group (object or group UUID) to add the entry to (default: root group)
+* C<group> (aka C<parent>) - Group object or group UUID to add the entry to (default: root group)
 
 =cut
 
@@ -596,7 +665,7 @@ sub add_entry {
 
     # find the right group to add the entry to
     my $parent = delete $args{group} // delete $args{parent} // $self->root;
-    ($parent) = $self->find_groups({uuid => $parent}) if !ref $parent;
+    $parent = $self->groups->grep({uuid => $parent})->next if !ref $parent;
     $parent or throw 'Invalid group';
 
     return $parent->add_entry(defined $entry ? $entry : (), %args, kdbx => $self);
@@ -609,97 +678,52 @@ sub _wrap_entry {
     return File::KDBX::Entry->wrap($entry, $self);
 }
 
-=method all_entries
+=method entries
 
-    \@entries = $kdbx->all_entries(%options);
-    \@entries = $kdbx->all_entries($base_group, %options);
+    \&iterator = $kdbx->entries(%options);
+    \&iterator = $kdbx->entries($base_group, %options);
 
-Get entries deeply in a database, in a flat array. Supported options:
+Get an iterator over I<entries> within a database. Supports the same options as L</groups>, plus some new
+ones:
 
 =for :list
-* C<base> - Only include entries within a base group (same as C<$base_group>) (default: root)
 * C<auto_type> - Only include entries with auto-type enabled (default: false, include all)
-* C<search> - Only include entries within groups with search enabled (default: false, include all)
-* C<history> - Also include historical entries (default: false, include only active entries)
+* C<searching> - Only include entries within groups with search enabled (default: false, include all)
+* C<history> - Also include historical entries (default: false, include only current entries)
 
 =cut
 
-sub all_entries {
+sub entries {
     my $self = shift;
     my %args = @_ % 2 == 0 ? @_ : (base => shift, @_);
+    my $base = delete $args{base} // $self->root;
 
-    my $base        = $args{base} // $self->root;
-    my $history     = $args{history};
-    my $search      = $args{search};
-    my $auto_type   = $args{auto_type};
-
-    my $enable_auto_type = $base->{enable_auto_type} // true;
-    my $enable_searching = $base->{enable_searching} // true;
-
-    my @entries;
-    if ((!$search || $enable_searching) && (!$auto_type || $enable_auto_type)) {
-        push @entries,
-            map { $self->_wrap_entry($_) }
-            grep { !$auto_type || $_->{auto_type}{enabled} }
-            map { $_, $history ? @{$_->{history} || []} : () }
-            @{$base->{entries} || []};
-    }
-
-    for my $subgroup (@{$base->{groups} || []}) {
-        my $more = $self->all_entries($subgroup,
-            auto_type   => $auto_type,
-            search      => $search,
-            history     => $history,
-        );
-        push @entries, @$more;
-    }
-
-    return \@entries;
+    return $base->entries_deeply(%args);
 }
 
-=method find_entries
-
-=method find_entries_simple
-
-    @entries = $kdbx->find_entries($query, %options);
+##############################################################################
 
-    @entries = $kdbx->find_entries_simple($expression, \@fields, %options);
-    @entries = $kdbx->find_entries_simple($expression, $operator, \@fields, %options);
+=method objects
 
-Find all entries deeply that match a query. Options are the same as for L</all_entries>.
+    \&iterator = $kdbx->objects(%options);
+    \&iterator = $kdbx->objects($base_group, %options);
 
-See L</QUERY> for a description of what C<$query> can be.
+Get an iterator over I<objects> within a database. Groups and entries are considered objects, so this is
+essentially a combination of L</groups> and L</entries>. This won't often be useful, but it can be convenient
+for maintenance tasks. This method takes the same options as L</groups> and L</entries>.
 
 =cut
 
-sub find_entries {
+sub objects {
     my $self = shift;
-    my $query = shift or throw 'Must provide a query';
-    my %args = @_;
-    my %all_entries = (
-        base        => $args{base},
-        auto_type   => $args{auto_type},
-        search      => $args{search},
-        history     => $args{history},
-    );
-    my $limit = delete $args{limit};
-    if (defined $limit) {
-        return @{search_limited($self->all_entries(%all_entries), is_arrayref($query) ? @$query : $query, $limit)};
-    }
-    else {
-        return @{search($self->all_entries(%all_entries), is_arrayref($query) ? @$query : $query)};
-    }
-}
+    my %args = @_ % 2 == 0 ? @_ : (base => shift, @_);
+    my $base = delete $args{base} // $self->root;
 
-sub find_entries_simple {
-    my $self = shift;
-    my $text = shift;
-    my $op   = @_ && !is_ref($_[0]) ? shift : undef;
-    my $fields = shift;
-    is_arrayref($fields) or throw q{Usage: find_entries_simple($expression, [$op,] \@fields)};
-    return $self->find_entries([\$text, $op, $fields], @_);
+    return $base->objects_deeply(%args);
 }
 
+sub __iter__ { $_[0]->objects }
+
 ##############################################################################
 
 =method custom_icon
@@ -709,46 +733,52 @@ sub find_entries_simple {
     $kdbx->custom_icon(%icon);
     $kdbx->custom_icon(uuid => $value, %icon);
 
+Get or set custom icons.
 
 =cut
 
 sub custom_icon {
     my $self = shift;
-    my %args = @_     == 2 ? (uuid => shift, value => shift)
+    my %args = @_     == 2 ? (uuid => shift, data => shift)
              : @_ % 2 == 1 ? (uuid => shift, @_) : @_;
 
-    if (!$args{key} && !$args{value}) {
-        my %standard = (key => 1, value => 1, last_modification_time => 1);
+    if (!$args{uuid} && !$args{data}) {
+        my %standard = (uuid => 1, data => 1, name => 1, last_modification_time => 1);
         my @other_keys = grep { !$standard{$_} } keys %args;
         if (@other_keys == 1) {
             my $key = $args{key} = $other_keys[0];
-            $args{value} = delete $args{$key};
+            $args{data} = delete $args{$key};
         }
     }
 
-    my $key = $args{key} or throw 'Must provide a custom_icons key to access';
+    my $uuid = $args{uuid} or throw 'Must provide a custom icon UUID to access';
+    my $icon = (first { $_->{uuid} eq $uuid } @{$self->custom_icons}) // do {
+        push @{$self->custom_icons}, my $i = { uuid => $uuid };
+        $i;
+    };
 
-    return $self->{meta}{custom_icons}{$key} = $args{value} if is_plain_hashref($args{value});
+    my $fields = \%args;
+    $fields = $args{data} if is_plain_hashref($args{data});
 
-    while (my ($field, $value) = each %args) {
-        $self->{meta}{custom_icons}{$key}{$field} = $value;
+    while (my ($field, $value) = each %$fields) {
+        $icon->{$field} = $value;
     }
-    return $self->{meta}{custom_icons}{$key};
+    return $icon;
 }
 
 =method custom_icon_data
 
     $image_data = $kdbx->custom_icon_data($uuid);
 
-Get a custom icon.
+Get a custom icon image data.
 
 =cut
 
 sub custom_icon_data {
     my $self = shift;
     my $uuid = shift // return;
-    return if !exists $self->custom_icons->{$uuid};
-    return $self->custom_icons->{$uuid}{data};
+    my $icon = first { $_->{uuid} eq $uuid } @{$self->custom_icons} or return;
+    return $icon->{data};
 }
 
 =method add_custom_icon
@@ -758,7 +788,7 @@ sub custom_icon_data {
 Add a custom icon and get its UUID. If not provided, a random UUID will be generated. Possible attributes:
 
 =for :list
-* C<uuid> - Icon UUID
+* C<uuid> - Icon UUID (default: autogenerated)
 * C<name> - Name of the icon (text, KDBX4.1+)
 * C<last_modification_time> - Just what it says (datetime, KDBX4.1+)
 
@@ -769,8 +799,8 @@ sub add_custom_icon {
     my $img  = shift or throw 'Must provide image data';
     my %args = @_;
 
-    my $uuid = $args{uuid} // generate_uuid(sub { !$self->custom_icons->{$_} });
-    $self->custom_icons->{$uuid} = {
+    my $uuid = $args{uuid} // generate_uuid;
+    push @{$self->custom_icons}, {
         @_,
         uuid    => $uuid,
         data    => $img,
@@ -789,7 +819,11 @@ Remove a custom icon.
 sub remove_custom_icon {
     my $self = shift;
     my $uuid = shift;
-    delete $self->custom_icons->{$uuid};
+    my @deleted;
+    @{$self->custom_icons} = grep { $_->{uuid} eq $uuid ? do { push @deleted, $_; 0 } : 1 }
+        @{$self->custom_icons};
+    $self->add_deleted_object($uuid) if @deleted;
+    return @deleted;
 }
 
 ##############################################################################
@@ -906,6 +940,59 @@ sub public_custom_data {
 #     die 'Not implemented';
 # }
 
+=method add_deleted_object
+
+    $kdbx->add_deleted_object($uuid);
+
+Add a UUID to the deleted objects list. This list is used to support automatic database merging.
+
+You typically do not need to call this yourself because the list will be populated automatically as objects
+are removed.
+
+=cut
+
+sub add_deleted_object {
+    my $self = shift;
+    my $uuid = shift;
+
+    # ignore null and meta stream UUIDs
+    return if $uuid eq UUID_NULL || $uuid eq '0' x 16;
+
+    $self->deleted_objects->{$uuid} = {
+        uuid            => $uuid,
+        deletion_time   => scalar gmtime,
+    };
+}
+
+=method remove_deleted_object
+
+    $kdbx->remove_deleted_object($uuid);
+
+Remove a UUID from the deleted objects list. This list is used to support automatic database merging.
+
+You typically do not need to call this yourself because the list will be maintained automatically as objects
+are added.
+
+=cut
+
+sub remove_deleted_object {
+    my $self = shift;
+    my $uuid = shift;
+    delete $self->deleted_objects->{$uuid};
+}
+
+=method clear_deleted_objects
+
+Remove all UUIDs from the deleted objects list.  This list is used to support automatic database merging, but
+if you don't need merging then you can clear deleted objects to reduce the database file size.
+
+=cut
+
+sub clear_deleted_objects {
+    my $self = shift;
+    %{$self->deleted_objects} = ();
+}
+
 ##############################################################################
 
 =method resolve_reference
@@ -919,8 +1006,8 @@ references are resolved automatically while expanding entry strings (i.e. replac
 use this method to resolve on-the-fly references that aren't part of any actual string in the database.
 
 If the reference does not resolve to any field, C<undef> is returned. If the reference resolves to multiple
-fields, only the first one is returned (in the same order as L</all_entries>). To avoid ambiguity, you can
-refer to a specific entry by its UUID.
+fields, only the first one is returned (in the same order as iterated by L</entries>). To avoid ambiguity, you
+can refer to a specific entry by its UUID.
 
 The syntax of a reference is: C<< {REF:<WantedField>@<SearchIn>:<Text>} >>. C<Text> is a
 L</"Simple Expression">. C<WantedField> and C<SearchIn> are both single character codes representing a field:
@@ -982,14 +1069,14 @@ sub resolve_reference {
     my $query = $search_in eq 'uuid' ? query($search_in => uuid($text))
                                      : simple_expression_query($text, '=~', $search_in);
 
-    my ($entry) = $self->find_entries($query, limit => 1);
+    my $entry = $self->entries->grep($query)->next;
     $entry or return;
 
     return $entry->$wanted;
 }
 
 our %PLACEHOLDERS = (
-    # placeholder         => sub { my ($entry, $arg) = @_; ... };
+    # 'PLACEHOLDER'       => sub { my ($entry, $arg) = @_; ... };
     'TITLE'             => sub { $_[0]->expanded_title },
     'USERNAME'          => sub { $_[0]->expanded_username },
     'PASSWORD'          => sub { $_[0]->expanded_password },
@@ -1017,9 +1104,9 @@ our %PLACEHOLDERS = (
     'OPERA'             => sub { load_optional('IPC::Cmd'); IPC::Cmd::can_run('opera') },
     'SAFARI'            => sub { load_optional('IPC::Cmd'); IPC::Cmd::can_run('safari') },
     'APPDIR'            => sub { load_optional('FindBin'); $FindBin::Bin },
-    'GROUP'             => sub { my $p = $_[0]->parent; $p ? $p->name : undef },
+    'GROUP'             => sub { my $p = $_[0]->group; $p ? $p->name : undef },
     'GROUP_PATH'        => sub { $_[0]->path },
-    'GROUP_NOTES'       => sub { my $p = $_[0]->parent; $p ? $p->notes : undef },
+    'GROUP_NOTES'       => sub { my $p = $_[0]->group; $p ? $p->notes : undef },
     # 'GROUP_SEL'
     # 'GROUP_SEL_PATH'
     # 'GROUP_SEL_NOTES'
@@ -1069,9 +1156,9 @@ our %PLACEHOLDERS = (
 
     $kdbx->lock;
 
-Encrypt all protected strings in a database. The encrypted strings are stored in a L<File::KDBX::Safe>
-associated with the database and the actual strings will be replaced with C<undef> to indicate their protected
-state. Returns itself to allow method chaining.
+Encrypt all protected binaries strings in a database. The encrypted strings are stored in
+a L<File::KDBX::Safe> associated with the database and the actual strings will be replaced with C<undef> to
+indicate their protected state. Returns itself to allow method chaining.
 
 =cut
 
@@ -1090,10 +1177,9 @@ sub lock {
 
     my @strings;
 
-    my $entries = $self->all_entries(history => 1);
-    for my $entry (@$entries) {
-        push @strings, grep { $_->{protect} } values %{$entry->{strings} || {}};
-    }
+    $self->entries(history => 1)->each(sub {
+        push @strings, grep { $_->{protect} } values %{$_->strings}, values %{$_->binaries};
+    });
 
     $self->_safe(File::KDBX::Safe->new(\@strings));
 
@@ -1207,11 +1293,11 @@ sub randomize_seeds {
     $key = $kdbx->key($key);
     $key = $kdbx->key($primitive);
 
-Get or set a L<File::KDBX::Key>. This is the master key (i.e. a password or a key file that can decrypt
+Get or set a L<File::KDBX::Key>. This is the master key (e.g. a password or a key file that can decrypt
 a database). See L<File::KDBX::Key/new> for an explanation of what the primitive can be.
 
 You generally don't need to call this directly because you can provide the key directly to the loader or
-dumper when loading or saving a KDBX file.
+dumper when loading or dumping a KDBX file.
 
 =cut
 
@@ -1382,7 +1468,7 @@ sub inner_random_stream_key {
 
 #########################################################################################
 
-sub check {
+sub check {
 # - Fixer tool. Can repair inconsistencies, including:
 #   - Orphaned binaries... not really a thing anymore since we now distribute binaries amongst entries
 #   - Unused custom icons (OFF, data loss)
@@ -1401,7 +1487,7 @@ sub check {
 #   - Duplicate window associations (OFF)
 #   - Only one root group (ON)
   # - Header UUIDs match known ciphers/KDFs?
-}
+}
 
 #########################################################################################
 
@@ -1411,35 +1497,38 @@ sub _handle_signal {
     my $type    = shift;
 
     my %handlers = (
-        'entry.uuid.changed'    => \&_update_entry_uuid,
-        'group.uuid.changed'    => \&_update_group_uuid,
+        'entry.added'           => \&_handle_object_added,
+        'group.added'           => \&_handle_object_added,
+        'entry.removed'         => \&_handle_object_removed,
+        'group.removed'         => \&_handle_object_removed,
+        'entry.uuid.changed'    => \&_handle_entry_uuid_changed,
+        'group.uuid.changed'    => \&_handle_group_uuid_changed,
     );
     my $handler = $handlers{$type} or return;
     $self->$handler($object, @_);
 }
 
-sub _update_group_uuid {
+sub _handle_object_added {
+    my $self    = shift;
+    my $object  = shift;
+    $self->remove_deleted_object($object->uuid);
+}
+
+sub _handle_object_removed {
     my $self        = shift;
     my $object      = shift;
-    my $new_uuid    = shift;
-    my $old_uuid    = shift // return;
+    my $old_uuid    = $object->{uuid} // return;
 
     my $meta = $self->meta;
-    $self->recycle_bin_uuid($new_uuid) if $old_uuid eq ($meta->{recycle_bin_uuid} // '');
-    $self->entry_templates_group($new_uuid) if $old_uuid eq ($meta->{entry_templates_group} // '');
-    $self->last_selected_group($new_uuid) if $old_uuid eq ($meta->{last_selected_group} // '');
-    $self->last_top_visible_group($new_uuid) if $old_uuid eq ($meta->{last_top_visible_group} // '');
-
-    for my $group (@{$self->all_groups}) {
-        $group->last_top_visible_entry($new_uuid) if $old_uuid eq ($group->{last_top_visible_entry} // '');
-        $group->previous_parent_group($new_uuid) if $old_uuid eq ($group->{previous_parent_group} // '');
-    }
-    for my $entry (@{$self->all_entries}) {
-        $entry->previous_parent_group($new_uuid) if $old_uuid eq ($entry->{previous_parent_group} // '');
-    }
+    $self->recycle_bin_uuid(UUID_NULL)          if $old_uuid eq ($meta->{recycle_bin_uuid} // '');
+    $self->entry_templates_group(UUID_NULL)     if $old_uuid eq ($meta->{entry_templates_group} // '');
+    $self->last_selected_group(UUID_NULL)       if $old_uuid eq ($meta->{last_selected_group} // '');
+    $self->last_top_visible_group(UUID_NULL)    if $old_uuid eq ($meta->{last_top_visible_group} // '');
+
+    $self->add_deleted_object($old_uuid);
 }
 
-sub _update_entry_uuid {
+sub _handle_entry_uuid_changed {
     my $self        = shift;
     my $object      = shift;
     my $new_uuid    = shift;
@@ -1449,16 +1538,37 @@ sub _update_entry_uuid {
     my $new_pretty = format_uuid($new_uuid);
     my $fieldref_match = qr/\{REF:([TUPANI])\@I:\Q$old_pretty\E\}/is;
 
-    for my $entry (@{$self->all_entries}) {
-        $entry->previous_parent_group($new_uuid) if $old_uuid eq ($entry->{previous_parent_group} // '');
+    $self->entries->each(sub {
+        $_->previous_parent_group($new_uuid) if $old_uuid eq ($_->{previous_parent_group} // '');
 
-        for my $string (values %{$entry->strings}) {
+        for my $string (values %{$_->strings}) {
             next if !defined $string->{value} || $string->{value} !~ $fieldref_match;
-            my $txn = $entry->begin_work;
+            my $txn = $_->begin_work;
             $string->{value} =~ s/$fieldref_match/{REF:$1\@I:$new_pretty}/g;
             $txn->commit;
         }
-    }
+    });
+}
+
+sub _handle_group_uuid_changed {
+    my $self        = shift;
+    my $object      = shift;
+    my $new_uuid    = shift;
+    my $old_uuid    = shift // return;
+
+    my $meta = $self->meta;
+    $self->recycle_bin_uuid($new_uuid)          if $old_uuid eq ($meta->{recycle_bin_uuid} // '');
+    $self->entry_templates_group($new_uuid)     if $old_uuid eq ($meta->{entry_templates_group} // '');
+    $self->last_selected_group($new_uuid)       if $old_uuid eq ($meta->{last_selected_group} // '');
+    $self->last_top_visible_group($new_uuid)    if $old_uuid eq ($meta->{last_top_visible_group} // '');
+
+    $self->groups->each(sub {
+        $_->last_top_visible_entry($new_uuid)   if $old_uuid eq ($_->{last_top_visible_entry} // '');
+        $_->previous_parent_group($new_uuid)    if $old_uuid eq ($_->{previous_parent_group} // '');
+    });
+    $self->entries->each(sub {
+        $_->previous_parent_group($new_uuid)    if $old_uuid eq ($_->{previous_parent_group} // '');
+    });
 }
 
 #########################################################################################
@@ -1672,9 +1782,10 @@ __END__
 
     $kdbx = File::KDBX->load_file('passwords.kdbx', 'M@st3rP@ssw0rd!');
 
-    for my $entry (@{ $kdbx->all_entries }) {
+    kdbx->entries->each(sub {
+        my ($entry) = @_;
         say 'Entry: ', $entry->title;
-    }
+    });
 
 =head1 DESCRIPTION
 
@@ -1730,32 +1841,50 @@ considerations.
     my $kdbx = File::KDBX->load_file('mypasswords.kdbx', 'master password CHANGEME');
     $kdbx->unlock;
 
-    for my $entry (@{ $kdbx->all_entries }) {
-        say 'Found password for ', $entry->title, ':';
+    $kdbx->entries->each(sub {
+        my ($entry) = @_;
+        say 'Found password for ', $entry->title;
         say '  Username: ', $entry->username;
         say '  Password: ', $entry->password;
-    }
+    });
 
 =head2 Search for entries
 
-    my @entries = $kdbx->find_entries({
-        title => 'WayneCorp',
-    }, search => 1);
+    my @entries = $kdbx->entries(searching => 1)
+        ->grep(title => 'WayneCorp')
+        ->each;     # return all matches
+
+The C<searching> option limits results to only entries within groups with searching enabled. Other options are
+also available. See L</entries>.
 
 See L</QUERY> for many more query examples.
 
 =head2 Search for entries by auto-type window association
 
-    my @entry_key_sequences = $kdbx->find_entries_for_window('WayneCorp - Mozilla Firefox');
-    for my $pair (@entry_key_sequences) {
-        my ($entry, $key_sequence) = @$pair;
-        say 'Entry title: ', $entry->title, ', key sequence: ', $key_sequence;
-    }
+    my $window_title = 'WayneCorp - Mozilla Firefox';
+
+    my $entries = $kdbx->entries(auto_type => 1)
+        ->filter(sub {
+            my ($ata) = grep { $_->{window} =~ /\Q$window_title\E/i } @{$_->auto_type_associations};
+            return [$_, $ata->{keystroke_sequence}] if $ata;
+        })
+        ->each(sub {
+            my ($entry, $keys) = @$_;
+            say 'Entry title: ', $entry->title, ', key sequence: ', $keys;
+        });
 
 Example output:
 
     Entry title: WayneCorp, key sequence: {PASSWORD}{ENTER}
 
+=head2 Remove entries from a database
+
+    $kdbx->entries
+        ->grep(notes => {'=~' => qr/too old/i})
+        ->each(sub { $_->recycle });
+
+Recycle all entries with the string "too old" appearing in the B<Notes> string.
+
 =head1 SECURITY
 
 One of the biggest threats to your database security is how easily the encryption key can be brute-forced.
@@ -1851,9 +1980,14 @@ unfortunately not portable.
 
 =head1 QUERY
 
-Several methods take a I<query> as an argument (e.g. L</find_entries>). A query is just a subroutine that you
-can either write yourself or have generated for you based on either a simple expression or a declarative
-structure. It's easier to have your query generated, so I'll cover that first.
+To find things in a KDBX database, you should use a filtered iterator. If you have an iterator, such as
+returned by L</entries>, L</groups> or even L</objects> you can filter it using L<File::KDBX::Iterator/where>.
+
+    my $filtered_results = $kdbx->entries->where($query);
+
+A C<$query> is just a subroutine that you can either write yourself or have generated for you from either
+a L</"Simple Expression"> or L</"Declarative Syntax">. It's easier to have your query generated, so I'll cover
+that first.
 
 =head2 Simple Expression
 
@@ -1866,55 +2000,56 @@ one of the given fields.
 
 So a simple expression is something like what you might type into a search engine. You can generate a simple
 expression query using L<File::KDBX::Util/simple_expression_query> or by passing the simple expression as
-a B<string reference> to search methods like L</find_entries>.
+a B<scalar reference> to C<where>.
 
 To search for all entries in a database with the word "canyon" appearing anywhere in the title:
 
-    my @entries = $kdbx->find_entries([ \'canyon', qw(title) ]);
+    my $entries = $kdbx->entries->where(\'canyon', qw[title]);
 
-Notice the first argument is a B<stringref>. This diambiguates a simple expression from other types of queries
+Notice the first argument is a B<scalarref>. This diambiguates a simple expression from other types of queries
 covered below.
 
 As mentioned, a simple expression can have multiple terms. This simple expression query matches any entry that
 has the words "red" B<and> "canyon" anywhere in the title:
 
-    my @entries = $kdbx->find_entries([ \'red canyon', qw(title) ]);
+    my $entries = $kdbx->entries->where(\'red canyon', qw[title]);
 
 Each term in the simple expression must be found for an entry to match.
 
 To search for entries with "red" in the title but B<not> "canyon", just prepend "canyon" with a minus sign:
 
-    my @entries = $kdbx->find_entries([ \'red -canyon', qw(title) ]);
+    my $entries = $kdbx->entries->where(\'red -canyon', qw[title]);
 
 To search over multiple fields simultaneously, just list them. To search for entries with "grocery" in the
 title or notes but not "Foodland":
 
-    my @entries = $kdbx->find_entries([ \'grocery -Foodland', qw(title notes) ]);
+    my $entries = $kdbx->entries->where(\'grocery -Foodland', qw[title notes]);
 
 The default operator is a case-insensitive regexp match, which is fine for searching text loosely. You can use
 just about any binary comparison operator that perl supports. To specify an operator, list it after the simple
 expression. For example, to search for any entry that has been used at least five times:
 
-    my @entries = $kdbx->find_entries([ \5, '>=', qw(usage_count) ]);
+    my $entries = $kdbx->entries->where(\5, '>=', qw[usage_count]);
 
 It helps to read it right-to-left, like "usage_count is >= 5".
 
-If you find the disambiguating structures to be confusing, you can also the L</find_entries_simple> method as
-a more intuitive alternative. The following example is equivalent to the previous:
+If you find the disambiguating structures to be distracting or confusing, you can also the
+L<File::KDBX::Util/simple_expression_query> function as a more intuitive alternative. The following example is
+equivalent to the previous:
 
-    my @entries = $kdbx->find_entries_simple(5, '>=', qw(usage_count));
+    my $entries = $kdbx->entries->where(simple_expression_query(5, '>=', qw[usage_count]));
 
-=head2 Declarative Query
+=head2 Declarative Syntax
 
 Structuring a declarative query is similar to L<SQL::Abstract/"WHERE CLAUSES">, but you don't have to be
 familiar with that module. Just learn by examples.
 
 To search for all entries in a database titled "My Bank":
 
-    my @entries = $kdbx->find_entries({ title => 'My Bank' });
+    my $entries = $kdbx->entries->where({ title => 'My Bank' });
 
-The query here is C<< { title => 'My Bank' } >>. A hashref can contain key-value pairs where the key is
-a attribute of the thing being searched for (in this case an entry) and the value is what you want the thing's
+The query here is C<< { title => 'My Bank' } >>. A hashref can contain key-value pairs where the key is an
+attribute of the thing being searched for (in this case an entry) and the value is what you want the thing's
 attribute to be to consider it a match. In this case, the attribute we're using as our match criteria is
 L<File::KDBX::Entry/title>, a text field. If an entry has its title attribute equal to "My Bank", it's
 a match.
@@ -1923,33 +2058,35 @@ A hashref can contain multiple attributes. The search candidate will be a match
 attributes are equal to their respective values. For example, to search for all entries with a particular URL
 B<AND> username:
 
-    my @entries = $kdbx->find_entries({
+    my $entries = $kdbx->entries->where({
         url      => 'https://example.com',
         username => 'neo',
     });
 
 To search for entries matching I<any> criteria, just change the hashref to an arrayref. To search for entries
-with a particular URL B<OR> a particular username:
+with a particular URL B<OR> username:
 
-    my @entries = $kdbx->find_entries([ # <-- square bracket
+    my $entries = $kdbx->entries->where([ # <-- Notice the square bracket
         url      => 'https://example.com',
         username => 'neo',
     ]);
 
-You can user different operators to test different types of attributes. The L<File::KDBX::Entry/icon_id>
+
+
+You can use different operators to test different types of attributes. The L<File::KDBX::Entry/icon_id>
 attribute is a number, so we should use a number comparison operator. To find entries using the smartphone
 icon:
 
-    my @entries = $kdbx->find_entries({
+    my $entries = $kdbx->entries->where({
         icon_id => { '==', ICON_SMARTPHONE },
     });
 
 Note: L<File::KDBX::Constants/ICON_SMARTPHONE> is just a constant from L<File::KDBX::Constants>. It isn't
 special to this example or to queries generally. We could have just used a literal number.
 
-The important thing to notice here is how we wrapped the condition in another arrayref with a single key-pair
-where the key is the name of an operator and the value is the thing to match against. The supported operators
-are:
+The important thing to notice here is how we wrapped the condition in another arrayref with a single key-value
+pair where the key is the name of an operator and the value is the thing to match against. The supported
+operators are:
 
 =for :list
 * C<eq> - String equal
@@ -1976,7 +2113,7 @@ Other special operators:
 * C<-false> - Boolean false
 * C<-not> - Boolean false (alias for C<-false>)
 * C<-defined> - Is defined
-* C<-undef> - Is not d efined
+* C<-undef> - Is not defined
 * C<-empty> - Is empty
 * C<-nonempty> - Is not empty
 * C<-or> - Logical or
@@ -1985,42 +2122,46 @@ Other special operators:
 Let's see another example using an explicit operator. To find all groups except one in particular (identified
 by its L<File::KDBX::Group/uuid>), we can use the C<ne> (string not equal) operator:
 
-    my ($group, @other) = $kdbx->find_groups({
+    my $groups = $kdbx->groups->where(
         uuid => {
             'ne' => uuid('596f7520-6172-6520-7370-656369616c2e'),
         },
-    });
-    if (@other) { say "Problem: there can be only one!" }
+    );
+    if (1 < $groups->count) { say "Problem: there can be only one!" }
 
-Note: L<File::KDBX::Util/uuid> is a little helper function to convert a UUID in its pretty form into octets.
+Note: L<File::KDBX::Util/uuid> is a little helper function to convert a UUID in its pretty form into bytes.
 This helper function isn't special to this example or to queries generally. It could have been written with
 a literal such as C<"\x59\x6f\x75\x20\x61...">, but that's harder to read.
 
 Notice we searched for groups this time. Finding groups works exactly the same as it does for entries.
 
+Notice also that we didn't wrap the query in hashref curly-braces or arrayref square-braces. Those are
+optional. By default it will only match ALL attributes (as if there were curly-braces), but it doesn't matter
+if there is only one attribute so it's fine to rely on the implicit behavior.
+
 Testing the truthiness of an attribute is a little bit different because it isn't a binary operation. To find
 all entries with the password quality check disabled:
 
-    my @entries = $kdbx->find_entries({ '!' => 'quality_check' });
+    my $entries = $kdbx->entries->where('!' => 'quality_check');
 
 This time the string after the operator is the attribute name rather than a value to compare the attribute
 against. To test that a boolean value is true, use the C<!!> operator (or C<-true> if C<!!> seems a little too
 weird for your taste):
 
-    my @entries = $kdbx->find_entries({ '!!'  => 'quality_check' });
-    my @entries = $kdbx->find_entries({ -true => 'quality_check' });
+    my $entries = $kdbx->entries->where('!!'  => 'quality_check');
+    my $entries = $kdbx->entries->where(-true => 'quality_check');
 
 Yes, there is also a C<-false> and a C<-not> if you prefer one of those over C<!>. C<-false> and C<-not>
 (along with C<-true>) are also special in that you can use them to invert the logic of a subquery. These are
 logically equivalent:
 
-    my @entries = $kdbx->find_entries([ -not => { title => 'My Bank' } ]);
-    my @entries = $kdbx->find_entries({ title => { 'ne' => 'My Bank' } });
+    my $entries = $kdbx->entries->where(-not => { title => 'My Bank' });
+    my $entries = $kdbx->entries->where(title => { 'ne' => 'My Bank' });
 
 These special operators become more useful when combined with two more special operators: C<-and> and C<-or>.
 With these, it is possible to construct more interesting queries with groups of logic. For example:
 
-    my @entries = $kdbx->find_entries({
+    my $entries = $kdbx->entries->where({
         title   => { '=~', qr/bank/ },
         -not    => {
             -or     => {
@@ -2031,22 +2172,20 @@ With these, it is possible to construct more interesting queries with groups of
     });
 
 In English, find entries where the word "bank" appears anywhere in the title but also do not have either the
-word "business" in the notes or is using the full trashcan icon.
+word "business" in the notes or are using the full trashcan icon.
 
 =head2 Subroutine Query
 
 Lastly, as mentioned at the top, you can ignore all this and write your own subroutine. Your subroutine will
-be called once for each thing being searched over. The single argument is the search candidate. The subroutine
-should match the candidate against whatever criteria you want and return true if it matches. The C<find_*>
-methods collect all matching things and return them.
+be called once for each object being searched over. The subroutine should match the candidate against whatever
+criteria you want and return true if it matches or false to skip. To do this, just pass your subroutine
+coderef to C<where>.
 
-For example, to find all entries in the database titled "My Bank":
+For example, these are all equivalent to find all entries in the database titled "My Bank":
 
-    my @entries = $kdbx->find_entries(sub { shift->title eq 'My Bank' });
-    # logically the same as this declarative structure:
-    my @entries = $kdbx->find_entries({ title => 'My Bank' });
-    # as well as this simple expression:
-    my @entries = $kdbx->find_entries([ \'My Bank', 'eq', qw{title} ]);
+    my $entries = $kdbx->entries->where(\'"My Bank"', 'eq', qw[title]);     # simple expression
+    my $entries = $kdbx->entries->where(title => 'My Bank');                # declarative syntax
+    my $entries = $kdbx->entries->where(sub { $_->title eq 'My Bank' });    # subroutine query
 
 This is a trivial example, but of course your subroutine can be arbitrarily complex.
 
@@ -2054,7 +2193,8 @@ All of these query mechanisms described in this section are just tools, each wit
 If the tools are getting in your way, you can of course iterate over the contents of a database and implement
 your own query logic, like this:
 
-    for my $entry (@{ $kdbx->all_entries }) {
+    my $entries = $kdbx->entries;
+    while (my $entry = $entries->next) {
         if (wanted($entry)) {
             do_something($entry);
         }
This page took 0.064122 seconds and 4 git commands to generate.