• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

realm / realm-core / 2131

13 Mar 2024 04:51AM UTC coverage: 91.785% (-0.05%) from 91.833%
2131

push

Evergreen

web-flow
Merge pull request #7402 from realm/tg/obj-perf

Make Obj trivial and add a separate ObjCollectionParent type

94394 of 174600 branches covered (54.06%)

496 of 559 new or added lines in 21 files covered. (88.73%)

224 existing lines in 28 files now uncovered.

242743 of 264469 relevant lines covered (91.79%)

5639637.18 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

88.89
/src/realm/sync/noinst/client_reset_operation.cpp
1
/*************************************************************************
2
 *
3
 * Copyright 2021 Realm Inc.
4
 *
5
 * Licensed under the Apache License, Version 2.0 (the "License");
6
 * you may not use this file except in compliance with the License.
7
 * You may obtain a copy of the License at
8
 *
9
 * http://www.apache.org/licenses/LICENSE-2.0
10
 *
11
 * Unless required by applicable law or agreed to in writing, software
12
 * distributed under the License is distributed on an "AS IS" BASIS,
13
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14
 * See the License for the specific language governing permissions and
15
 * limitations under the License.
16
 *
17
 **************************************************************************/
18

19
#include <realm/sync/noinst/client_reset_operation.hpp>
20

21
#include <realm/sync/history.hpp>
22
#include <realm/sync/noinst/client_history_impl.hpp>
23
#include <realm/sync/noinst/client_reset.hpp>
24
#include <realm/transaction.hpp>
25
#include <realm/util/scope_exit.hpp>
26

27
namespace realm::_impl::client_reset {
28

29
namespace {
30

31
constexpr static std::string_view c_fresh_suffix(".fresh");
32

33
} // namespace
34

35
std::string get_fresh_path_for(const std::string& path)
36
{
245,548✔
37
    const size_t suffix_len = c_fresh_suffix.size();
245,548✔
38
    REALM_ASSERT(path.length());
245,548✔
39
    REALM_ASSERT_DEBUG_EX(
245,548✔
40
        path.size() < suffix_len || path.substr(path.size() - suffix_len, suffix_len) != c_fresh_suffix, path);
245,548✔
41
    return path + c_fresh_suffix.data();
245,548✔
42
}
245,548✔
43

44
bool is_fresh_path(const std::string& path)
45
{
3,776✔
46
    const size_t suffix_len = c_fresh_suffix.size();
3,776✔
47
    REALM_ASSERT(path.length());
3,776✔
48
    if (path.size() < suffix_len) {
3,776✔
49
        return false;
×
50
    }
×
51
    return path.substr(path.size() - suffix_len, suffix_len) == c_fresh_suffix;
3,776✔
52
}
3,776✔
53

54
bool perform_client_reset(util::Logger& logger, DB& db, DB& fresh_db, ClientResyncMode mode,
55
                          CallbackBeforeType notify_before, CallbackAfterType notify_after,
56
                          sync::SaltedFileIdent new_file_ident, sync::SubscriptionStore* sub_store,
57
                          util::FunctionRef<void(int64_t)> on_flx_version, bool recovery_is_allowed)
58
{
460✔
59
    REALM_ASSERT(mode != ClientResyncMode::Manual);
460✔
60
    logger.debug(util::LogCategory::reset,
460✔
61
                 "Possibly beginning client reset operation: realm_path = %1, mode = %2, recovery_allowed = %3",
460✔
62
                 db.get_path(), mode, recovery_is_allowed);
460✔
63

230✔
64
    auto always_try_clean_up = util::make_scope_exit([&]() noexcept {
460✔
65
        std::string path_to_clean = fresh_db.get_path();
460✔
66
        try {
460✔
67
            fresh_db.close();
460✔
68
            constexpr bool delete_lockfile = true;
460✔
69
            DB::delete_files(path_to_clean, nullptr, delete_lockfile);
460✔
70
        }
460✔
71
        catch (const std::exception& err) {
230✔
UNCOV
72
            logger.warn(util::LogCategory::reset,
×
UNCOV
73
                        "In ClientResetOperation::finalize, the fresh copy '%1' could not be cleaned up due to "
×
UNCOV
74
                        "an exception: '%2'",
×
UNCOV
75
                        path_to_clean, err.what());
×
76
            // ignored, this is just a best effort
UNCOV
77
        }
×
78
    });
460✔
79

230✔
80
    // only do the reset if there is data to reset
230✔
81
    // if there is nothing in this Realm, then there is nothing to reset and
230✔
82
    // sync should be able to continue as normal
230✔
83
    auto latest_version = db.get_version_id_of_latest_snapshot();
460✔
84
    bool local_realm_exists = latest_version.version > 1;
460✔
85
    if (!local_realm_exists) {
460✔
86
        logger.debug(util::LogCategory::reset,
4✔
87
                     "Local Realm file has never been written to, so skipping client reset.");
4✔
88
        return false;
4✔
89
    }
4✔
90

228✔
91
    VersionID frozen_before_state_version = notify_before ? notify_before() : latest_version;
456✔
92

228✔
93
    // If m_notify_after is set, pin the previous state to keep it around.
228✔
94
    TransactionRef previous_state;
456✔
95
    if (notify_after) {
456✔
96
        previous_state = db.start_frozen(frozen_before_state_version);
260✔
97
    }
260✔
98
    bool did_recover = client_reset::perform_client_reset_diff(
456✔
99
        db, fresh_db, new_file_ident, logger, mode, recovery_is_allowed, sub_store, on_flx_version); // throws
456✔
100

228✔
101
    if (notify_after) {
456✔
102
        notify_after(previous_state->get_version_of_current_transaction(), did_recover);
200✔
103
    }
200✔
104

228✔
105
    return true;
456✔
106
}
456✔
107

108
} // namespace realm::_impl::client_reset
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc