GCC Code Coverage Report


Directory: ./
File: pdcom5/src/msrproto/DirNode.cpp
Date: 2025-01-19 04:08:20
Exec Total Coverage
Lines: 79 81 97.5%
Branches: 104 152 68.4%

Line Branch Exec Source
1 /*****************************************************************************
2 *
3 * Copyright (C) 2015-2016 Richard Hacker (lerichi at gmx dot net)
4 *
5 * This file is part of the PdCom library.
6 *
7 * The PdCom library is free software: you can redistribute it and/or modify
8 * it under the terms of the GNU Lesser General Public License as published by
9 * the Free Software Foundation, either version 3 of the License, or (at your
10 * option) any later version.
11 *
12 * The PdCom library is distributed in the hope that it will be useful, but
13 * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
14 * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
15 * License for more details.
16 *
17 * You should have received a copy of the GNU Lesser General Public License
18 * along with the PdCom library. If not, see <http://www.gnu.org/licenses/>.
19 *
20 *****************************************************************************/
21
22 #include "DirNode.h"
23
24 #include "../Debug.h"
25
26 #include <algorithm> // std::copy
27 #include <cstring> // strchr()
28 #include <iterator> // back_inserter
29
30 using namespace PdCom::impl::MsrProto;
31
32 namespace {
33 ///////////////////////////////////////////////////////////////////////////
34 ///////////////////////////////////////////////////////////////////////////
35 class DirectoryPathIterator
36 {
37 public:
38 960 DirectoryPathIterator(const char *path) { pathPtr = path; }
39
40 3202 std::string next()
41 {
42 3202 const char *p = pathPtr;
43
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 3202 times.
3202 pathPtr = strchr(pathPtr, dirSeparator);
44
12/22
✓ Branch 1 taken 2451 times.
✓ Branch 2 taken 751 times.
✓ Branch 8 taken 2451 times.
✗ Branch 9 not taken.
✓ Branch 13 taken 751 times.
✗ Branch 14 not taken.
✓ Branch 15 taken 751 times.
✓ Branch 16 taken 2451 times.
✓ Branch 18 taken 751 times.
✓ Branch 19 taken 2451 times.
✓ Branch 21 taken 2451 times.
✓ Branch 22 taken 751 times.
✓ Branch 24 taken 2451 times.
✓ Branch 25 taken 751 times.
✗ Branch 27 not taken.
✗ Branch 28 not taken.
✗ Branch 30 not taken.
✗ Branch 31 not taken.
✗ Branch 33 not taken.
✗ Branch 34 not taken.
✗ Branch 36 not taken.
✗ Branch 37 not taken.
3202 return pathPtr ? std::string(p, skip() - p) : p;
45 }
46
47
3/4
✓ Branch 1 taken 2820 times.
✓ Branch 2 taken 474 times.
✓ Branch 5 taken 2820 times.
✗ Branch 6 not taken.
3294 bool hasNext() const { return pathPtr and *pathPtr; }
48
49 private:
50 const char *pathPtr;
51
52 static const char dirSeparator;
53
54 2451 const char *skip()
55 {
56 2451 const char *p = pathPtr;
57
58
3/4
✓ Branch 2 taken 4902 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 2451 times.
✓ Branch 7 taken 2451 times.
7353 while (*pathPtr and *pathPtr == dirSeparator)
59 2451 ++pathPtr;
60
61 2451 return p;
62 }
63 };
64
65 } // namespace
66 const char DirectoryPathIterator::dirSeparator = '/';
67
68 ///////////////////////////////////////////////////////////////////////////
69 ///////////////////////////////////////////////////////////////////////////
70 709 DirNode::DirNode(bool isDir)
71 {
72 709 m_parent = this;
73 709 directory = isDir;
74 709 }
75
76 ///////////////////////////////////////////////////////////////////////////
77 1779 std::string DirNode::path() const
78 {
79
13/24
✓ Branch 3 taken 481 times.
✓ Branch 4 taken 1298 times.
✓ Branch 13 taken 1298 times.
✗ Branch 14 not taken.
✓ Branch 17 taken 1298 times.
✗ Branch 18 not taken.
✓ Branch 22 taken 1298 times.
✗ Branch 23 not taken.
✓ Branch 24 taken 1298 times.
✓ Branch 25 taken 481 times.
✓ Branch 27 taken 1298 times.
✓ Branch 28 taken 481 times.
✓ Branch 30 taken 1298 times.
✓ Branch 31 taken 481 times.
✓ Branch 33 taken 1298 times.
✓ Branch 34 taken 481 times.
✗ Branch 36 not taken.
✗ Branch 37 not taken.
✗ Branch 39 not taken.
✗ Branch 40 not taken.
✗ Branch 42 not taken.
✗ Branch 43 not taken.
✗ Branch 45 not taken.
✗ Branch 46 not taken.
1779 return m_parent == this ? std::string() : (m_parent->path() + '/' + m_name);
80 }
81
82 ///////////////////////////////////////////////////////////////////////////
83 382 void DirNode::insert(std::shared_ptr<DirNode> child, const char *cpath)
84 {
85 382 DirectoryPathIterator path(cpath);
86
1/2
✓ Branch 2 taken 382 times.
✗ Branch 3 not taken.
764 std::shared_ptr<DirNode> parent = shared_from_this();
87
88
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 382 times.
382 if (!path.hasNext()) {
89 return;
90 }
91
92 while (true) {
93
4/5
✓ Branch 2 taken 1436 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 672 times.
✓ Branch 7 taken 382 times.
✓ Branch 8 taken 382 times.
2108 std::string name = path.next();
94
95
2/2
✓ Branch 1 taken 382 times.
✓ Branch 2 taken 1054 times.
1818 if (name.empty()) {
96 // Find root node
97
1/2
✓ Branch 2 taken 382 times.
✗ Branch 3 not taken.
382 parent = getRootNode();
98 382 continue;
99 }
100
101 1054 NodeVector::iterator it = std::lower_bound(
102 3162 parent->children.begin(), parent->children.end(), name,
103
4/5
✓ Branch 4 taken 1054 times.
✗ Branch 5 not taken.
✓ Branch 9 taken 672 times.
✓ Branch 10 taken 382 times.
✓ Branch 11 taken 382 times.
4888 LessThan());
104
105 // Insert a new node if it exists
106
8/10
✓ Branch 5 taken 617 times.
✓ Branch 6 taken 437 times.
✓ Branch 12 taken 113 times.
✓ Branch 13 taken 504 times.
✓ Branch 14 taken 1054 times.
✗ Branch 15 not taken.
✓ Branch 17 taken 1054 times.
✗ Branch 18 not taken.
✓ Branch 20 taken 550 times.
✓ Branch 21 taken 504 times.
1054 if (it == parent->children.end() or (*it)->m_name != name)
107
1/2
✓ Branch 8 taken 550 times.
✗ Branch 9 not taken.
550 it = parent->children.insert(it, 0);
108 1054 std::shared_ptr<DirNode> &node = *it;
109
110
2/2
✓ Branch 1 taken 672 times.
✓ Branch 2 taken 382 times.
1054 if (path.hasNext()) {
111
2/2
✓ Branch 2 taken 170 times.
✓ Branch 3 taken 502 times.
672 if (!node) {
112
2/4
✓ Branch 2 taken 170 times.
✗ Branch 3 not taken.
✓ Branch 7 taken 170 times.
✗ Branch 8 not taken.
170 node.reset(new DirNode);
113 170 node->m_parent = parent.get();
114
1/2
✓ Branch 4 taken 170 times.
✗ Branch 5 not taken.
170 node->m_name = name;
115 }
116 672 parent = node;
117 }
118 else {
119
2/2
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 380 times.
382 if (node) {
120 // Node exists. This can happen, when a deep node has
121 // been discovered before the current node. An example is a
122 // signal that has parameters as children
123 2 std::swap(node->children, child->children);
124 2 child->directory = true;
125
2/2
✓ Branch 7 taken 2 times.
✓ Branch 8 taken 2 times.
4 for (auto &cchild : child->children)
126 2 cchild->m_parent = child.get();
127
128 2 node.reset();
129 }
130
131 382 child->m_parent = parent.get();
132
1/2
✓ Branch 3 taken 382 times.
✗ Branch 4 not taken.
382 child->m_name = name;
133 382 node = std::move(child);
134
135
2/2
✓ Branch 1 taken 672 times.
✓ Branch 2 taken 382 times.
382 return;
136 }
137 1054 }
138 }
139
140 ///////////////////////////////////////////////////////////////////////////
141 5 std::string DirNode::name() const
142 {
143 5 return m_name;
144 }
145
146 ///////////////////////////////////////////////////////////////////////////
147 578 std::shared_ptr<DirNode> DirNode::find(const std::string &pathStr)
148 {
149
1/2
✓ Branch 2 taken 578 times.
✗ Branch 3 not taken.
1156 std::shared_ptr<DirNode> node = shared_from_this();
150 578 DirectoryPathIterator path(pathStr.c_str());
151
152
2/2
✓ Branch 1 taken 1766 times.
✓ Branch 2 taken 92 times.
3138 while (path.hasNext()) {
153
4/5
✓ Branch 2 taken 1766 times.
✗ Branch 3 not taken.
✓ Branch 6 taken 702 times.
✓ Branch 7 taken 578 times.
✓ Branch 8 taken 486 times.
2468 std::string name = path.next();
154
155
2/2
✓ Branch 1 taken 578 times.
✓ Branch 2 taken 1188 times.
2344 if (name.empty()) {
156 // Find root node
157
1/2
✓ Branch 2 taken 578 times.
✗ Branch 3 not taken.
578 node = getRootNode();
158 578 continue;
159 }
160
161 1188 NodeVector::iterator it = std::lower_bound(
162
4/5
✓ Branch 8 taken 1188 times.
✗ Branch 9 not taken.
✓ Branch 16 taken 702 times.
✓ Branch 17 taken 578 times.
✓ Branch 18 taken 486 times.
1890 node->children.begin(), node->children.end(), name, LessThan());
163
164
8/10
✓ Branch 5 taken 816 times.
✓ Branch 6 taken 372 times.
✓ Branch 12 taken 114 times.
✓ Branch 13 taken 702 times.
✓ Branch 14 taken 1188 times.
✗ Branch 15 not taken.
✓ Branch 17 taken 1188 times.
✗ Branch 18 not taken.
✓ Branch 20 taken 486 times.
✓ Branch 21 taken 702 times.
1188 if (it == node->children.end() or name != (*it)->m_name)
165 486 return nullptr;
166
167
2/2
✓ Branch 4 taken 702 times.
✓ Branch 5 taken 486 times.
702 node = *it;
168 }
169
170 92 return node;
171 }
172
173 ///////////////////////////////////////////////////////////////////////////
174 12 void DirNode::getChildren(List &list) const
175 {
176
2/2
✓ Branch 6 taken 18 times.
✓ Branch 7 taken 12 times.
30 for (const auto &child : children)
177
1/2
✓ Branch 3 taken 18 times.
✗ Branch 4 not taken.
18 list.push_back(child);
178 12 }
179
180 ///////////////////////////////////////////////////////////////////////////
181 264 void DirNode::getAllChildren(List &list) const
182 {
183
2/2
✓ Branch 6 taken 258 times.
✓ Branch 7 taken 264 times.
522 for (const auto &child : children) {
184
1/2
✓ Branch 3 taken 258 times.
✗ Branch 4 not taken.
258 list.push_back(child);
185
1/2
✓ Branch 6 taken 258 times.
✗ Branch 7 not taken.
258 child->getAllChildren(list);
186 }
187 264 }
188
189 ///////////////////////////////////////////////////////////////////////////
190 18 bool DirNode::hasChildren() const
191 {
192
4/6
✗ Branch 3 not taken.
✓ Branch 4 taken 18 times.
✓ Branch 5 taken 12 times.
✓ Branch 6 taken 6 times.
✗ Branch 9 not taken.
✓ Branch 10 taken 12 times.
18 return directory or !children.empty();
193 }
194
195 /////////////////////////////////////////////////////////////////////////////
196 960 std::shared_ptr<DirNode> DirNode::getRootNode()
197 {
198 960 DirNode *n = this;
199
1/2
✗ Branch 3 not taken.
✓ Branch 4 taken 960 times.
960 while (n->m_parent != n)
200 n = n->m_parent;
201 960 return n->shared_from_this();
202 }
203