Sat, 01 Feb 2025 16:40:24 +0100
remove -s flag which is unknown to clang
14 | 1 | /* Copyright 2025 Mike Becker. All rights reserved. |
2 | * | |
3 | * Redistribution and use in source and binary forms, with or without | |
4 | * modification, are permitted provided that the following conditions are met: | |
5 | * | |
6 | * 1. Redistributions of source code must retain the above copyright | |
7 | * notice, this list of conditions and the following disclaimer. | |
8 | * | |
9 | * 2. Redistributions in binary form must reproduce the above copyright | |
10 | * notice, this list of conditions and the following disclaimer in the | |
11 | * documentation and/or other materials provided with the distribution. | |
12 | * | |
13 | * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | |
14 | * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | |
15 | * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |
16 | * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | |
17 | * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | |
18 | * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | |
19 | * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | |
20 | * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | |
21 | * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |
22 | * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |
23 | */ | |
24 | ||
25 | #include "settings.h" | |
26 | ||
27 | #include <format> | |
28 | #include <algorithm> | |
16 | 29 | #include <fstream> |
14 | 30 | |
31 | using namespace fm; | |
32 | ||
16 | 33 | static bool check_author(std::string_view author, std::string_view allowed) { |
34 | // check for exact match | |
35 | if (author == allowed) return true; | |
36 | ||
37 | // check mail address matching | |
38 | if (author.contains(std::format("<{}>", allowed))) return true; | |
39 | ||
40 | // check local-part from mail address matching | |
41 | if (author.contains(std::format("<{}@", allowed))) return true; | |
42 | ||
43 | return false; | |
44 | } | |
45 | ||
14 | 46 | [[nodiscard]] bool settings::exclude_author(const std::string &author) const { |
47 | // no allow-list means: always allowed | |
48 | if (authors.empty()) return false; | |
49 | ||
50 | // check all allowed authors | |
16 | 51 | return !std::ranges::any_of(authors, [&](const auto &allowed) { |
52 | return check_author(author, allowed); | |
14 | 53 | }); |
54 | } | |
16 | 55 | |
56 | std::string_view trim(const std::string& str) { | |
57 | size_t s = str.find_first_not_of(" \t"); | |
58 | size_t l = str.find_last_not_of(" \t") + 1 - s; | |
59 | return std::string_view{str}.substr(s, l); | |
60 | } | |
61 | ||
62 | int settings::parse_authormap(const std::string& path) { | |
63 | std::ifstream file(path); | |
64 | ||
65 | if (!file.is_open()) { | |
66 | return -1; | |
67 | } | |
68 | ||
69 | std::string line; | |
70 | while (std::getline(file, line)) { | |
71 | line = trim(line); | |
72 | ||
73 | // skip empty lines and comments | |
74 | if (line.empty() || line[0] == '#') { | |
75 | continue; | |
76 | } | |
77 | ||
78 | // find delimiter | |
79 | size_t delimPos = line.find('='); | |
80 | if (delimPos == std::string::npos) { | |
81 | return -1; | |
82 | } | |
83 | ||
84 | auto key = std::string{trim(line.substr(0, delimPos))}; | |
85 | auto value = std::string{trim(line.substr(delimPos + 1))}; | |
86 | authormap[std::move(key)] = std::move(value); | |
87 | } | |
88 | ||
89 | return 0; | |
90 | } | |
91 | ||
92 | std::string settings::map_author(std::string_view author) const { | |
93 | if (authormap.empty()) return std::string{author}; | |
94 | ||
95 | for (const auto &[old_name, new_name] : authormap) { | |
96 | if (check_author(author, old_name)) return new_name; | |
97 | } | |
98 | ||
99 | return std::string{author}; | |
100 | } |