0.6.1
C++ to UML diagram generator based on Clang
Loading...
Searching...
No Matches
util.h
Go to the documentation of this file.
1/**
2 * @file src/util/util.h
3 *
4 * Copyright (c) 2021-2025 Bartek Kryza <bkryza@gmail.com>
5 *
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
17 */
18#pragma once
19
20#include "logging.h"
21
22#include <algorithm>
23#include <cstring>
24#include <filesystem>
25#include <map>
26#include <optional>
27#include <string>
28#include <type_traits>
29#include <vector>
30
31namespace clanguml::util {
32
33constexpr unsigned kDefaultMessageCommentWidth{25U};
34constexpr unsigned kDefaultMessageNameWidth{100U};
35
36/**
37 * @brief Left trim a string
38 *
39 * @param s Input string
40 * @return Left trimmed string
41 */
42std::string ltrim(const std::string &s);
43
44/**
45 * @brief Right trim a string
46 *
47 * @param s Input string
48 * @return Right trimmed string
49 */
50std::string rtrim(const std::string &s);
51
52/**
53 * @brief Trim a string
54 *
55 * @param s Input string
56 * @return Trimmed string
57 */
58std::string trim(const std::string &s);
59
60/**
61 * @brief Remove `typename` prefix from a string if exists
62 * @param s Input string
63 * @return String without `typename` prefix
64 */
65std::string trim_typename(const std::string &s);
66
67/**
68 * @brief Execute a shell `command` and return console output as string
69 *
70 * @param command Shell command to execute
71 * @return Console output of the command
72 */
73std::string get_process_output(const std::string &command);
74
75/**
76 * @brief Execute command shell and throw exception if command fails
77 *
78 * @param command Command to execute
79 */
80void check_process_output(const std::string &command);
81
82/**
83 * @brief Get value of an environment variable
84 *
85 * @param name Name of the environment variable
86 * @return Value of the environment variable, or empty if it doesn't exist
87 */
88std::string get_env(const std::string &name);
89
90/**
91 * @brief Check if `$PWD` is in a Git repository
92 *
93 * This can be overridden by exporting `CLANGUML_GIT_COMMIT` environment
94 * variable.
95 *
96 * @return True, if the current directory is in a Git repository
97 */
99
100/**
101 * @brief Get current Git branch
102 *
103 * @return Name of the current Git branch
104 */
105std::string get_git_branch();
106
107/**
108 * @brief Get current Git revision
109 *
110 * Generates a Git revision tag using `git describe --tags --always` command
111 *
112 * @return Current repository Git revision
113 */
114std::string get_git_revision();
115
116/**
117 * @brief Get current Git commit
118 *
119 * @return Latest Git commit hash
120 */
121std::string get_git_commit();
122
123/**
124 * @brief Get path to the top level Git directory
125 *
126 * @return Absolut path to the nearest directory containing `.git` folder
127 */
128std::string get_git_toplevel_dir();
129
130/**
131 * @brief Get descriptive name of the current operating system.
132 *
133 * @return Name of the operating system
134 */
135std::string get_os_name();
136
137template <typename T, typename S>
138std::unique_ptr<T> unique_pointer_cast(std::unique_ptr<S> &&p) noexcept
139{
140 if (T *const converted = dynamic_cast<T *>(p.get())) {
141 std::move(p).release(); // NOLINT
142 return std::unique_ptr<T>{converted};
143 }
144
145 return {};
146}
147
148/**
149 * @brief Split string at first occurence of separator
150 * @param separator
151 * @param input
152 * @return Elements from
153 */
154std::optional<std::pair<std::string, std::string>> split_at_first(
155 const std::string &separator, const std::string &input);
156
157/**
158 * @brief Split a string using delimiter
159 *
160 * Basic string split function, because C++ stdlib does not have one.
161 * In case the string does not contain the delimiter, the original
162 * string is returned as the only element of the vector.
163 *
164 * @param str String to split
165 * @param delimiter Delimiter string
166 * @param skip_empty Skip empty toks between delimiters if true
167 *
168 * @return Vector of string tokens.
169 */
170std::vector<std::string> split(
171 std::string str, std::string_view delimiter, bool skip_empty = true);
172
173std::vector<std::string> split_isspace(std::string str);
174
175/**
176 * @brief Remove and erase elements from a vector
177 *
178 * @tparam T Element type
179 * @tparam F Functor type
180 * @param v Vector to remove elements from
181 * @param f Functor to decide which elements to remove
182 */
183template <typename T, typename F> void erase_if(std::vector<T> &v, F &&f)
184{
185 v.erase(std::remove_if(v.begin(), v.end(), std::forward<F>(f)), v.end());
186}
187
188/**
189 * @brief Join `toks` into string using `delimiter` as separator
190 *
191 * @param toks Elements to join into string
192 * @param delimiter Separator to use to join elements
193 * @return Concatenated elements into one string
194 */
195std::string join(
196 const std::vector<std::string> &toks, std::string_view delimiter);
197
198/**
199 * @brief Join `args` into string using `delimiter` as separator
200 *
201 * @tparam Args Element type
202 * @param delimiter Separator to use to join elements
203 * @param args Elements to join into string
204 * @return Arguments concatenated into one string
205 */
206template <typename... Args>
207std::string join(std::string_view delimiter, Args... args)
208{
209 std::vector<std::string> coll{args...};
210
211 erase_if(coll, [](const auto &s) {
212 return s.find_first_not_of(" \t") == std::string::npos;
213 });
214
215 return fmt::format("{}", fmt::join(coll, delimiter));
216}
217
218/**
219 * @brief Abbreviate string to max_length, and replace last 3 characters
220 * with ellipsis.
221 *
222 * @param s Input string
223 * @param max_length Maximum length
224 * @return Abbreviated string
225 */
226std::string abbreviate(const std::string &s, unsigned int max_length);
227
228/**
229 * @brief Find element alias in Puml note
230 *
231 * Finds aliases of the form @A(entity_name) in the Puml notes
232 * or directives.
233 * The match, if any, is returned in the result tuple:
234 * (entity_name, offset, length)
235 *
236 * @return True if match was found
237 */
239 const std::string &input, std::tuple<std::string, size_t, size_t> &result);
240
241/**
242 * @brief Find and replace in string
243 *
244 * Replaces all occurences of pattern with replace_with in input string.
245 *
246 * @return True if at least on replacement was made
247 */
248bool replace_all(std::string &input, const std::string &pattern,
249 const std::string &replace_with);
250
251/**
252 * Replace all consecutive space like characters with just one space
253 *
254 * @param input
255 * @return Condensed string
256 */
257std::string condense_whitespace(const std::string &input);
258
259/**
260 * @brief Appends a vector to a vector.
261 *
262 * @tparam T
263 * @param l
264 * @param r
265 */
266template <typename T> void append(std::vector<T> &l, const std::vector<T> &r)
267{
268 l.insert(l.end(), r.begin(), r.end());
269}
270
271/**
272 * @brief Checks if collection starts with a prefix.
273 *
274 * @tparam T e.g. std::vector<std::string>
275 * @param col Collection to be checked against prefix
276 * @param prefix Container, which specifies the prefix
277 * @return true if first prefix.size() elements of col are equal to prefix
278 */
279template <typename T> bool starts_with(const T &col, const T &prefix)
280{
281 if (prefix.size() > col.size())
282 return false;
283
284 return std::search(col.begin(), col.end(), prefix.begin(), prefix.end()) ==
285 col.begin();
286}
287
288template <> bool starts_with(const std::string &s, const std::string &prefix);
289
290template <typename T> bool ends_with(const T &value, const T &suffix);
291
292template <> bool ends_with(const std::string &value, const std::string &suffix);
293
294template <typename T>
295bool ends_with(const std::vector<T> &col, const std::vector<T> &suffix)
296{
297 if (suffix.size() > col.size())
298 return false;
299
300 return std::vector<std::string>(suffix.rbegin(), suffix.rend()) ==
301 std::vector<std::string>(col.rbegin(), col.rbegin() + suffix.size());
302}
303
304/**
305 * @brief Removes prefix sequence of elements from the beginning of col.
306 *
307 * @tparam T
308 * @param col
309 * @param prefix
310 */
311template <typename T>
312void remove_prefix(std::vector<T> &col, const std::vector<T> &prefix)
313{
314 if (!starts_with(col, prefix))
315 return;
316
317 col = std::vector<T>(col.begin() + prefix.size(), col.end());
318}
319
320/**
321 * Returns true if element exists in container.
322 *
323 * @tparam T
324 * @tparam E
325 * @param container
326 * @param element
327 * @return
328 */
329template <typename T, typename E>
330bool contains(const T &container, const E &element)
331{
332 if constexpr (std::is_pointer_v<E>) {
333 return std::find_if(container.begin(), container.end(),
334 [&element](const auto &e) { return *e == *element; }) !=
335 container.end();
336 }
337 else if constexpr (std::is_same_v<std::remove_cv_t<T>, std::string>) {
338 return container.find(element) != std::string::npos;
339 }
340 else {
341 return std::find(container.begin(), container.end(), element) !=
342 container.end();
343 }
344}
345
346template <typename T, typename F> void for_each(const T &collection, F &&func)
347{
348 std::for_each(std::begin(collection), std::end(collection),
349 std::forward<decltype(func)>(func));
350}
351
352template <typename R, typename T, typename F>
353std::vector<R> map(const std::vector<T> &in, F &&f)
354{
355 std::vector<R> out;
356 std::transform(
357 in.cbegin(), in.cend(), std::back_inserter(out), std::forward<F>(f));
358 return out;
359}
360
361template <typename T, typename F, typename FElse>
362void if_not_null(const T *pointer, F &&func, FElse &&func_else)
363{
364 if (pointer != nullptr) {
365 std::forward<F>(func)(pointer);
366 }
367 else {
368 std::forward<FElse>(func_else)();
369 }
370}
371
372template <typename T, typename F> void if_not_null(const T *pointer, F &&func)
373{
374 if_not_null(pointer, std::forward<F>(func), []() {});
375}
376
377template <typename F, typename FElse>
378void _if(const bool condition, F &&func, FElse &&func_else)
379{
380 if (condition) {
381 std::forward<F>(func)();
382 }
383 else {
384 std::forward<FElse>(func_else)();
385 }
386}
387
388template <typename F> void _if(const bool condition, F &&func)
389{
390 _if(condition, std::forward<F>(func), []() {});
391}
392
393template <typename T> void remove_duplicates(T &coll)
394{
395 std::sort(coll.begin(), coll.end());
396 coll.erase(std::unique(coll.begin(), coll.end()), coll.end());
397}
398
399/**
400 * @brief Generate a hash seed.
401 *
402 * @param seed Initial seed.
403 * @return Hash seed.
404 */
405std::size_t hash_seed(std::size_t seed);
406
407/**
408 * @brief Convert filesystem path to url path
409 *
410 * The purpose of this function is to make sure that a path can
411 * be used in a URL, e.g. it's separators are POSIX-style.
412 *
413 * @param p Path to convert
414 * @return String representation of the path in URL format
415 */
416std::string path_to_url(const std::filesystem::path &p);
417
418/**
419 * @brief Ensure path is absolute.
420 *
421 * If path is absolute, return the p. If path is not absolute, make it
422 * absolute with respect to root directory.
423 *
424 * @param p Path to modify
425 * @param root Root against which the path should be made absolute
426 * @return Absolute path
427 */
428std::filesystem::path ensure_path_is_absolute(const std::filesystem::path &p,
429 const std::filesystem::path &root = std::filesystem::current_path());
430
431/**
432 * @brief Check if a given path is relative to another path.
433 *
434 * @param parent The path to be checked for relativity.
435 * @param right The base path against which the relativity is checked.
436 * @return True if the child path is relative to the parent path, false
437 * otherwise.
438 */
439bool is_relative_to(
440 const std::filesystem::path &parent, const std::filesystem::path &child);
441
442std::string format_message_comment(
443 const std::string &c, unsigned width = kDefaultMessageCommentWidth);
444
445std::optional<std::pair<std::string, std::string>> find_entry_by_path_prefix(
446 const std::map<std::string, std::string> &m, const std::string &prefix);
447} // namespace clanguml::util