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

realm / realm-core / 1776

20 Oct 2023 07:34PM UTC coverage: 91.609% (+0.04%) from 91.567%
1776

push

Evergreen

web-flow
Added sync socket result enum for sync socket callback handlers in C API (#7015)

* Added sync socket result enum for sync socket callback handlers in C API
* Updated changelog
* CAPI: timer callbacks are now released by canceled/complete function
* Updated c_api tests to use all sync socket c_api functions
* Additional updates to sync socket c api test
* Added CAPI write callback manager to manage async write callbacks
* Pass error codes up to default socket provider for async_write_binary() callbacks
* Removed async write callback manager from CAPI
* Updated changelog after release
* clang format and updates from review
* Update async_write_binary() error handling to not throw exception
* Updated a few comments
* Another comment update
* Updates from review

94360 of 173622 branches covered (0.0%)

111 of 150 new or added lines in 8 files covered. (74.0%)

59 existing lines in 18 files now uncovered.

230702 of 251832 relevant lines covered (91.61%)

6635584.64 hits per line

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

87.78
/src/realm/array_string.cpp
1
/*************************************************************************
2
 *
3
 * Copyright 2016 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/array_string.hpp>
20
#include <realm/spec.hpp>
21
#include <realm/mixed.hpp>
22

23
using namespace realm;
24

25
ArrayString::ArrayString(Allocator& a)
26
    : m_alloc(a)
27
{
20,079,813✔
28
    m_arr = new (&m_storage.m_string_short) ArrayStringShort(a, true);
20,079,813✔
29
}
20,079,813✔
30

31
void ArrayString::create()
32
{
363,420✔
33
    static_cast<ArrayStringShort*>(m_arr)->create();
363,420✔
34
}
363,420✔
35

36
void ArrayString::init_from_mem(MemRef mem) noexcept
37
{
18,083,589✔
38
    char* header = mem.get_addr();
18,083,589✔
39

9,032,529✔
40
    ArrayParent* parent = m_arr->get_parent();
18,083,589✔
41
    size_t ndx_in_parent = m_arr->get_ndx_in_parent();
18,083,589✔
42

9,032,529✔
43
    bool long_strings = Array::get_hasrefs_from_header(header);
18,083,589✔
44
    if (!long_strings) {
18,083,589✔
45
        // Small strings
6,716,745✔
46
        bool is_small = Array::get_wtype_from_header(header) == Array::wtype_Multiply;
13,452,960✔
47
        if (is_small) {
13,452,960✔
48
            auto arr = new (&m_storage.m_string_short) ArrayStringShort(m_alloc, m_nullable);
7,780,941✔
49
            arr->init_from_mem(mem);
7,780,941✔
50
            m_type = Type::small_strings;
7,780,941✔
51
        }
7,780,941✔
52
        else {
5,672,019✔
53
            auto arr = new (&m_storage.m_enum) Array(m_alloc);
5,672,019✔
54
            arr->init_from_mem(mem);
5,672,019✔
55
            m_string_enum_values = std::make_unique<ArrayString>(m_alloc);
5,672,019✔
56
            ArrayParent* p;
5,672,019✔
57
            REALM_ASSERT(m_spec != nullptr);
5,672,019✔
58
            REALM_ASSERT(m_col_ndx != realm::npos);
5,672,019✔
59
            ref_type r = m_spec->get_enumkeys_ref(m_col_ndx, p);
5,672,019✔
60
            m_string_enum_values->init_from_ref(r);
5,672,019✔
61
            m_string_enum_values->set_parent(p, m_col_ndx);
5,672,019✔
62
            m_type = Type::enum_strings;
5,672,019✔
63
        }
5,672,019✔
64
    }
13,452,960✔
65
    else {
4,630,629✔
66
        bool is_big = Array::get_context_flag_from_header(header);
4,630,629✔
67
        if (!is_big) {
4,630,629✔
68
            auto arr = new (&m_storage.m_string_long) ArraySmallBlobs(m_alloc);
1,209,507✔
69
            arr->init_from_mem(mem);
1,209,507✔
70
            m_type = Type::medium_strings;
1,209,507✔
71
        }
1,209,507✔
72
        else {
3,421,122✔
73
            auto arr = new (&m_storage.m_big_blobs) ArrayBigBlobs(m_alloc, m_nullable);
3,421,122✔
74
            arr->init_from_mem(mem);
3,421,122✔
75
            m_type = Type::big_strings;
3,421,122✔
76
        }
3,421,122✔
77
    }
4,630,629✔
78
    m_arr->set_parent(parent, ndx_in_parent);
18,083,589✔
79
}
18,083,589✔
80

81
void ArrayString::init_from_parent()
82
{
5,327,448✔
83
    ref_type ref = m_arr->get_ref_from_parent();
5,327,448✔
84
    init_from_ref(ref);
5,327,448✔
85
}
5,327,448✔
86

87
void ArrayString::destroy() noexcept
88
{
366✔
89
    if (m_arr->is_attached()) {
366✔
90
        Array::destroy_deep(m_arr->get_ref(), m_alloc);
360✔
91
        detach();
360✔
92
    }
360✔
93
}
366✔
94

95
void ArrayString::detach() noexcept
96
{
281,490✔
97
    m_arr->detach();
281,490✔
98
    // Make sure the object is in a state like right after construction
140,307✔
99
    // Next call must be to create()
140,307✔
100
    m_arr = new (&m_storage.m_string_short) ArrayStringShort(m_alloc, true);
281,490✔
101
    m_type = Type::small_strings;
281,490✔
102
}
281,490✔
103

104
size_t ArrayString::size() const
105
{
15,865,248✔
106
    switch (m_type) {
15,865,248✔
107
        case Type::small_strings:
6,765,618✔
108
            return static_cast<ArrayStringShort*>(m_arr)->size();
6,765,618✔
109
        case Type::medium_strings:
425,796✔
110
            return static_cast<ArraySmallBlobs*>(m_arr)->size();
425,796✔
111
        case Type::big_strings:
8,681,616✔
112
            return static_cast<ArrayBigBlobs*>(m_arr)->size();
8,681,616✔
113
        case Type::enum_strings:
282✔
114
            return static_cast<Array*>(m_arr)->size();
282✔
115
    }
×
116
    return {};
×
117
}
×
118

119
void ArrayString::add(StringData value)
120
{
160,410✔
121
    switch (upgrade_leaf(value.size())) {
160,410✔
122
        case Type::small_strings:
104,589✔
123
            static_cast<ArrayStringShort*>(m_arr)->add(value);
104,589✔
124
            break;
104,589✔
125
        case Type::medium_strings:
27,633✔
126
            static_cast<ArraySmallBlobs*>(m_arr)->add_string(value);
27,633✔
127
            break;
27,633✔
128
        case Type::big_strings:
28,191✔
129
            static_cast<ArrayBigBlobs*>(m_arr)->add_string(value);
28,191✔
130
            break;
28,191✔
131
        case Type::enum_strings: {
✔
132
            auto a = static_cast<Array*>(m_arr);
×
133
            size_t ndx = a->size();
×
134
            a->add(0);
×
135
            set(ndx, value);
×
136
            break;
×
137
        }
×
138
    }
160,410✔
139
}
160,410✔
140

141
void ArrayString::set(size_t ndx, StringData value)
142
{
3,811,059✔
143
    switch (upgrade_leaf(value.size())) {
3,811,059✔
144
        case Type::small_strings:
531,594✔
145
            static_cast<ArrayStringShort*>(m_arr)->set(ndx, value);
531,594✔
146
            break;
531,594✔
147
        case Type::medium_strings:
246,324✔
148
            static_cast<ArraySmallBlobs*>(m_arr)->set_string(ndx, value);
246,324✔
149
            break;
246,324✔
150
        case Type::big_strings:
409,059✔
151
            static_cast<ArrayBigBlobs*>(m_arr)->set_string(ndx, value);
409,059✔
152
            break;
409,059✔
153
        case Type::enum_strings: {
2,625,570✔
154
            size_t sz = m_string_enum_values->size();
2,625,570✔
155
            size_t res = m_string_enum_values->find_first(value, 0, sz);
2,625,570✔
156
            if (res == realm::not_found) {
2,625,570✔
157
                m_string_enum_values->add(value);
9,738✔
158
                res = sz;
9,738✔
159
            }
9,738✔
160
            static_cast<Array*>(m_arr)->set(ndx, res);
2,625,570✔
161
            break;
2,625,570✔
162
        }
×
163
    }
3,811,059✔
164
}
3,811,059✔
165

166
void ArrayString::insert(size_t ndx, StringData value)
167
{
3,350,052✔
168
    switch (upgrade_leaf(value.size())) {
3,350,052✔
169
        case Type::small_strings:
1,014,636✔
170
            static_cast<ArrayStringShort*>(m_arr)->insert(ndx, value);
1,014,636✔
171
            break;
1,014,636✔
172
        case Type::medium_strings:
285,426✔
173
            static_cast<ArraySmallBlobs*>(m_arr)->insert_string(ndx, value);
285,426✔
174
            break;
285,426✔
175
        case Type::big_strings:
738,297✔
176
            static_cast<ArrayBigBlobs*>(m_arr)->insert_string(ndx, value);
738,297✔
177
            break;
738,297✔
178
        case Type::enum_strings: {
1,312,722✔
179
            static_cast<Array*>(m_arr)->insert(ndx, 0);
1,312,722✔
180
            set(ndx, value);
1,312,722✔
181
        }
1,312,722✔
182
    }
3,350,052✔
183
}
3,350,052✔
184

185
StringData ArrayString::get(size_t ndx) const
186
{
22,802,343✔
187
    switch (m_type) {
22,802,343✔
188
        case Type::small_strings:
9,207,075✔
189
            return static_cast<ArrayStringShort*>(m_arr)->get(ndx);
9,207,075✔
190
        case Type::medium_strings:
1,356,561✔
191
            return static_cast<ArraySmallBlobs*>(m_arr)->get_string(ndx);
1,356,561✔
192
        case Type::big_strings:
9,302,727✔
193
            return static_cast<ArrayBigBlobs*>(m_arr)->get_string(ndx);
9,302,727✔
194
        case Type::enum_strings: {
3,073,917✔
195
            size_t index = size_t(static_cast<Array*>(m_arr)->get(ndx));
3,073,917✔
196
            return m_string_enum_values->get(index);
3,073,917✔
197
        }
×
198
    }
×
199
    return {};
×
200
}
×
201

202
StringData ArrayString::get_legacy(size_t ndx) const
203
{
6,648✔
204
    switch (m_type) {
6,648✔
205
        case Type::small_strings:
402✔
206
            return static_cast<ArrayStringShort*>(m_arr)->get(ndx);
402✔
207
        case Type::medium_strings:
3,060✔
208
            return static_cast<ArraySmallBlobs*>(m_arr)->get_string_legacy(ndx);
3,060✔
209
        case Type::big_strings:
3,078✔
210
            return static_cast<ArrayBigBlobs*>(m_arr)->get_string(ndx);
3,078✔
211
        case Type::enum_strings: {
108✔
212
            size_t index = size_t(static_cast<Array*>(m_arr)->get(ndx));
108✔
213
            return m_string_enum_values->get(index);
108✔
214
        }
×
215
    }
×
216
    return {};
×
217
}
×
218

219
Mixed ArrayString::get_any(size_t ndx) const
220
{
22,464✔
221
    return Mixed(get(ndx));
22,464✔
222
}
22,464✔
223

224
bool ArrayString::is_null(size_t ndx) const
225
{
4,138,869✔
226
    switch (m_type) {
4,138,869✔
227
        case Type::small_strings:
711,810✔
228
            return static_cast<ArrayStringShort*>(m_arr)->is_null(ndx);
711,810✔
229
        case Type::medium_strings:
11,685✔
230
            return static_cast<ArraySmallBlobs*>(m_arr)->is_null(ndx);
11,685✔
231
        case Type::big_strings:
3,420,696✔
232
            return static_cast<ArrayBigBlobs*>(m_arr)->is_null(ndx);
3,420,696✔
233
        case Type::enum_strings: {
✔
234
            size_t index = size_t(static_cast<Array*>(m_arr)->get(ndx));
×
235
            return m_string_enum_values->is_null(index);
×
236
        }
×
237
    }
×
238
    return {};
×
239
}
×
240

241
void ArrayString::erase(size_t ndx)
242
{
181,167✔
243
    switch (m_type) {
181,167✔
244
        case Type::small_strings:
90,117✔
245
            static_cast<ArrayStringShort*>(m_arr)->erase(ndx);
90,117✔
246
            break;
90,117✔
247
        case Type::medium_strings:
58,659✔
248
            static_cast<ArraySmallBlobs*>(m_arr)->erase(ndx);
58,659✔
249
            break;
58,659✔
250
        case Type::big_strings:
24,036✔
251
            static_cast<ArrayBigBlobs*>(m_arr)->erase(ndx);
24,036✔
252
            break;
24,036✔
253
        case Type::enum_strings:
8,361✔
254
            static_cast<Array*>(m_arr)->erase(ndx);
8,361✔
255
            break;
8,361✔
256
    }
181,167✔
257
}
181,167✔
258

259
void ArrayString::move(ArrayString& dst, size_t ndx)
260
{
66✔
261
    size_t sz = size();
66✔
262
    for (size_t i = ndx; i < sz; i++) {
20,730✔
263
        dst.add(get(i));
20,664✔
264
    }
20,664✔
265

33✔
266
    switch (m_type) {
66✔
267
        case Type::small_strings:
✔
268
            static_cast<ArrayStringShort*>(m_arr)->truncate(ndx);
×
269
            break;
×
270
        case Type::medium_strings:
60✔
271
            static_cast<ArraySmallBlobs*>(m_arr)->truncate(ndx);
60✔
272
            break;
60✔
273
        case Type::big_strings:
6✔
274
            static_cast<ArrayBigBlobs*>(m_arr)->truncate(ndx);
6✔
275
            break;
6✔
276
        case Type::enum_strings:
✔
277
            // this operation will never be called for enumerated columns
278
            REALM_UNREACHABLE();
×
279
            break;
×
280
    }
66✔
281
}
66✔
282

283
void ArrayString::clear()
284
{
417✔
285
    switch (m_type) {
417✔
286
        case Type::small_strings:
363✔
287
            static_cast<ArrayStringShort*>(m_arr)->clear();
363✔
288
            break;
363✔
289
        case Type::medium_strings:
6✔
290
            static_cast<ArraySmallBlobs*>(m_arr)->clear();
6✔
291
            break;
6✔
292
        case Type::big_strings:
48✔
293
            static_cast<ArrayBigBlobs*>(m_arr)->clear();
48✔
294
            break;
48✔
295
        case Type::enum_strings:
✔
296
            static_cast<Array*>(m_arr)->clear();
×
297
            break;
×
298
    }
417✔
299
}
417✔
300

301
size_t ArrayString::find_first(StringData value, size_t begin, size_t end) const noexcept
302
{
4,080,465✔
303
    switch (m_type) {
4,080,465✔
304
        case Type::small_strings:
3,122,817✔
305
            return static_cast<ArrayStringShort*>(m_arr)->find_first(value, begin, end);
3,122,817✔
306
        case Type::medium_strings: {
105,336✔
307
            BinaryData as_binary(value.data(), value.size());
105,336✔
308
            return static_cast<ArraySmallBlobs*>(m_arr)->find_first(as_binary, true, begin, end);
105,336✔
309
            break;
×
310
        }
×
311
        case Type::big_strings: {
254,259✔
312
            BinaryData as_binary(value.data(), value.size());
254,259✔
313
            return static_cast<ArrayBigBlobs*>(m_arr)->find_first(as_binary, true, begin, end);
254,259✔
314
            break;
×
315
        }
×
316
        case Type::enum_strings: {
598,392✔
317
            size_t sz = m_string_enum_values->size();
598,392✔
318
            size_t res = m_string_enum_values->find_first(value, 0, sz);
598,392✔
319
            if (res != realm::not_found) {
598,392✔
320
                return static_cast<Array*>(m_arr)->find_first(res, begin, end);
598,392✔
321
            }
598,392✔
UNCOV
322
            break;
×
UNCOV
323
        }
×
324
    }
×
325
    return not_found;
×
326
}
×
327

328
namespace {
329

330
template <class T>
331
inline StringData get_string(const T* arr, size_t ndx)
332
{
437,337✔
333
    return arr->get_string(ndx);
437,337✔
334
}
437,337✔
335

336
template <>
337
inline StringData get_string(const ArrayStringShort* arr, size_t ndx)
338
{
93,678✔
339
    return arr->get(ndx);
93,678✔
340
}
93,678✔
341

342
template <class T, class U>
343
size_t lower_bound_string(const T* arr, U value)
344
{
234,912✔
345
    size_t i = 0;
234,912✔
346
    size_t sz = arr->size();
234,912✔
347
    while (0 < sz) {
765,927✔
348
        size_t half = sz / 2;
531,015✔
349
        size_t mid = i + half;
531,015✔
350
        auto probe = get_string(arr, mid);
531,015✔
351
        if (probe < value) {
531,015✔
352
            i = mid + 1;
190,854✔
353
            sz -= half + 1;
190,854✔
354
        }
190,854✔
355
        else {
340,161✔
356
            sz = half;
340,161✔
357
        }
340,161✔
358
    }
531,015✔
359
    return i;
234,912✔
360
}
234,912✔
361
}
362

363
size_t ArrayString::lower_bound(StringData value)
364
{
234,912✔
365
    switch (m_type) {
234,912✔
366
        case Type::small_strings:
58,434✔
367
            return lower_bound_string(static_cast<ArrayStringShort*>(m_arr), value);
58,434✔
368
        case Type::medium_strings:
68,418✔
369
            return lower_bound_string(static_cast<ArraySmallBlobs*>(m_arr), value);
68,418✔
370
        case Type::big_strings:
108,060✔
371
            return lower_bound_string(static_cast<ArrayBigBlobs*>(m_arr), value);
108,060✔
372
        case Type::enum_strings:
✔
373
            break;
×
374
    }
×
375
    return realm::npos;
×
376
}
×
377

378
ArrayString::Type ArrayString::upgrade_leaf(size_t value_size)
379
{
7,318,317✔
380
    if (m_type == Type::big_strings)
7,318,317✔
381
        return Type::big_strings;
1,137,894✔
382

3,036,780✔
383
    if (m_type == Type::enum_strings)
6,180,423✔
384
        return Type::enum_strings;
3,938,265✔
385

1,067,751✔
386
    if (m_type == Type::medium_strings) {
2,242,158✔
387
        if (value_size <= medium_string_max_size)
530,367✔
388
            return Type::medium_strings;
529,845✔
389

282✔
390
        // Upgrade root leaf from medium to big strings
282✔
391
        auto string_medium = static_cast<ArraySmallBlobs*>(m_arr);
522✔
392
        ArrayBigBlobs big_blobs(m_alloc, true);
522✔
393
        big_blobs.create(); // Throws
522✔
394

282✔
395
        size_t n = string_medium->size();
522✔
396
        for (size_t i = 0; i < n; i++) {
10,926✔
397
            big_blobs.add_string(string_medium->get_string(i)); // Throws
10,404✔
398
        }
10,404✔
399
        auto parent = string_medium->get_parent();
522✔
400
        auto ndx_in_parent = string_medium->get_ndx_in_parent();
522✔
401
        string_medium->destroy();
522✔
402

282✔
403
        auto arr = new (&m_storage.m_big_blobs) ArrayBigBlobs(m_alloc, true);
522✔
404
        arr->init_from_mem(big_blobs.get_mem());
522✔
405
        arr->set_parent(parent, ndx_in_parent);
522✔
406
        arr->update_parent();
522✔
407

282✔
408
        m_type = Type::big_strings;
522✔
409
        return Type::big_strings;
522✔
410
    }
522✔
411

802,152✔
412
    // m_type == Type::small
802,152✔
413
    if (value_size <= small_string_max_size)
1,711,791✔
414
        return Type::small_strings;
1,650,441✔
415

30,108✔
416
    if (value_size <= medium_string_max_size) {
61,350✔
417
        // Upgrade root leaf from small to medium strings
14,502✔
418
        auto string_short = static_cast<ArrayStringShort*>(m_arr);
29,520✔
419
        ArraySmallBlobs string_long(m_alloc);
29,520✔
420
        string_long.create(); // Throws
29,520✔
421

14,502✔
422
        size_t n = string_short->size();
29,520✔
423
        for (size_t i = 0; i < n; i++) {
63,957✔
424
            string_long.add_string(string_short->get(i)); // Throws
34,437✔
425
        }
34,437✔
426
        auto parent = string_short->get_parent();
29,520✔
427
        auto ndx_in_parent = string_short->get_ndx_in_parent();
29,520✔
428
        string_short->destroy();
29,520✔
429

14,502✔
430
        auto arr = new (&m_storage.m_string_long) ArraySmallBlobs(m_alloc);
29,520✔
431
        arr->init_from_mem(string_long.get_mem());
29,520✔
432
        arr->set_parent(parent, ndx_in_parent);
29,520✔
433
        arr->update_parent();
29,520✔
434

14,502✔
435
        m_type = Type::medium_strings;
29,520✔
436
    }
29,520✔
437
    else {
31,830✔
438
        // Upgrade root leaf from small to big strings
15,606✔
439
        auto string_short = static_cast<ArrayStringShort*>(m_arr);
31,830✔
440
        ArrayBigBlobs big_blobs(m_alloc, true);
31,830✔
441
        big_blobs.create(); // Throws
31,830✔
442

15,606✔
443
        size_t n = string_short->size();
31,830✔
444
        for (size_t i = 0; i < n; i++) {
69,459✔
445
            big_blobs.add_string(string_short->get(i)); // Throws
37,629✔
446
        }
37,629✔
447
        auto parent = string_short->get_parent();
31,830✔
448
        auto ndx_in_parent = string_short->get_ndx_in_parent();
31,830✔
449
        string_short->destroy();
31,830✔
450

15,606✔
451
        auto arr = new (&m_storage.m_big_blobs) ArrayBigBlobs(m_alloc, true);
31,830✔
452
        arr->init_from_mem(big_blobs.get_mem());
31,830✔
453
        arr->set_parent(parent, ndx_in_parent);
31,830✔
454
        arr->update_parent();
31,830✔
455

15,606✔
456
        m_type = Type::big_strings;
31,830✔
457
    }
31,830✔
458

30,108✔
459
    return m_type;
61,350✔
460
}
61,350✔
461

462
void ArrayString::verify() const
463
{
2,972,403✔
464
#ifdef REALM_DEBUG
2,972,403✔
465
    switch (m_type) {
2,972,403✔
466
        case Type::small_strings:
2,965,935✔
467
            static_cast<ArrayStringShort*>(m_arr)->verify();
2,965,935✔
468
            break;
2,965,935✔
469
        case Type::medium_strings:
711✔
470
            static_cast<ArraySmallBlobs*>(m_arr)->verify();
711✔
471
            break;
711✔
472
        case Type::big_strings:
5,511✔
473
            static_cast<ArrayBigBlobs*>(m_arr)->verify();
5,511✔
474
            break;
5,511✔
475
        case Type::enum_strings:
246✔
476
            static_cast<Array*>(m_arr)->verify();
246✔
477
            break;
246✔
478
    }
2,972,403✔
479
#endif
2,972,403✔
480
}
2,972,403✔
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

© 2025 Coveralls, Inc