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

randombit / botan / 23225340130

18 Mar 2026 01:53AM UTC coverage: 89.677% (-0.001%) from 89.678%
23225340130

push

github

web-flow
Merge pull request #5456 from randombit/jack/clang-tidy-22

Fix various warnings from clang-tidy 22

104438 of 116460 relevant lines covered (89.68%)

11819947.55 hits per line

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

89.08
/src/tests/test_roughtime.cpp
1
/*
2
* (C) 2019 Nuno Goncalves <nunojpg@gmail.com>
3
*
4
* Botan is released under the Simplified BSD License (see license.txt)
5
*/
6

7
#include "tests.h"
8

9
#if defined(BOTAN_HAS_BIGINT)
10
   #include <botan/bigint.h>
11
#endif
12

13
#if defined(BOTAN_HAS_ROUGHTIME)
14
   #include <botan/base64.h>
15
   #include <botan/ed25519.h>
16
   #include <botan/mem_ops.h>
17
   #include <botan/rng.h>
18
   #include <botan/roughtime.h>
19
#endif
20

21
namespace Botan_Tests {
22

23
namespace {
24

25
#if defined(BOTAN_HAS_ROUGHTIME)
26

27
class Roughtime_Request_Tests final : public Text_Based_Test {
×
28
   public:
29
      Roughtime_Request_Tests() : Text_Based_Test("roughtime/roughtime_request.vec", "Nonce,Request") {}
2✔
30

31
      Test::Result run_one_test(const std::string& type, const VarMap& vars) override {
2✔
32
         Test::Result result("Roughtime request");
2✔
33

34
         const auto nonce = Botan::Roughtime::Nonce(vars.get_req_bin("Nonce"));
2✔
35
         const auto request_v = vars.get_req_bin("Request");
2✔
36

37
         const auto request = Botan::Roughtime::encode_request(nonce);
2✔
38

39
         // TODO should use byte comparison function
40
         if(type == "Valid") {
2✔
41
            result.test_is_true("encode", request == Botan::typecast_copy<std::array<uint8_t, 1024>>(request_v.data()));
1✔
42
         } else {
43
            result.test_is_true("encode", request != Botan::typecast_copy<std::array<uint8_t, 1024>>(request_v.data()));
1✔
44
         }
45

46
         return result;
2✔
47
      }
2✔
48
};
49

50
BOTAN_REGISTER_TEST("roughtime", "roughtime_request", Roughtime_Request_Tests);
51

52
class Roughtime_Response_Tests final : public Text_Based_Test {
×
53
   public:
54
      Roughtime_Response_Tests() :
1✔
55
            Text_Based_Test(
56
               "roughtime/roughtime_response.vec", "Response", "Nonce,Pubkey,MidpointMicroSeconds,RadiusMicroSeconds") {
2✔
57
      }
1✔
58

59
      Test::Result run_one_test(const std::string& type, const VarMap& vars) override {
17✔
60
         Test::Result result("Roughtime response");
17✔
61

62
         const auto response_v = vars.get_req_bin("Response");
17✔
63
         const auto nonce_bits = vars.has_key("Nonce") ? vars.get_opt_bin("Nonce") : std::vector<uint8_t>(64);
17✔
64

65
         const Botan::Roughtime::Nonce nonce(nonce_bits);
17✔
66
         try {
17✔
67
            const auto response = Botan::Roughtime::Response::from_bits(response_v, nonce);
17✔
68

69
            const auto pubkey = vars.get_req_bin("Pubkey");
4✔
70
            if(pubkey.size() != 32) {
4✔
71
               throw Test_Error("Unexpected Roughtime Ed25519 pubkey size");
×
72
            }
73

74
            if(!response.validate(Botan::Ed25519_PublicKey(pubkey))) {
4✔
75
               result.test_is_true("fail_validation", type == "Invalid");
1✔
76
            } else {
77
               const auto midpoint = Botan::Roughtime::Response::sys_microseconds64(
3✔
78
                  std::chrono::microseconds(vars.get_req_u64("MidpointMicroSeconds")));
3✔
79
               const auto radius = std::chrono::microseconds(vars.get_req_u32("RadiusMicroSeconds"));
3✔
80

81
               result.test_is_true("midpoint", response.utc_midpoint() == midpoint);
3✔
82
               result.test_is_true("radius", response.utc_radius() == radius);
3✔
83
               result.test_is_true("OK", type == "Valid");
3✔
84
            }
85
         } catch(const Botan::Roughtime::Roughtime_Error& e) {
17✔
86
            result.test_is_true(e.what(), type == "Invalid");
13✔
87
         }
13✔
88

89
         return result;
17✔
90
      }
34✔
91
};
92

93
BOTAN_REGISTER_TEST("roughtime", "roughtime_response", Roughtime_Response_Tests);
94

95
class Roughtime_nonce_from_blind_Tests final : public Text_Based_Test {
×
96
   public:
97
      Roughtime_nonce_from_blind_Tests() :
1✔
98
            Text_Based_Test("roughtime/roughtime_nonce_from_blind.vec", "Response,Blind,Nonce") {}
2✔
99

100
      Test::Result run_one_test(const std::string& type, const VarMap& vars) override {
2✔
101
         Test::Result result("roughtime nonce_from_blind");
2✔
102

103
         const auto response = vars.get_req_bin("Response");
2✔
104
         const auto blind = Botan::Roughtime::Nonce(vars.get_req_bin("Blind"));
2✔
105
         const auto nonce = Botan::Roughtime::Nonce(vars.get_req_bin("Nonce"));
2✔
106

107
         const auto from_blind = Botan::Roughtime::nonce_from_blind(response, blind);
2✔
108

109
         if(type == "Valid") {
2✔
110
            result.test_is_true("valid nonce_from_blind", nonce == from_blind);
1✔
111
         } else {
112
            result.test_is_false("valid nonce_from_blind", nonce == from_blind);
1✔
113
         }
114

115
         return result;
2✔
116
      }
2✔
117
};
118

119
BOTAN_REGISTER_TEST("roughtime", "roughtime_nonce_from_blind", Roughtime_nonce_from_blind_Tests);
120

121
class Roughtime final : public Test {
1✔
122
   private:
123
      static Test::Result test_nonce(Botan::RandomNumberGenerator& rng) {
1✔
124
         Test::Result result("roughtime nonce");
1✔
125

126
         auto rand64 = Botan::unlock(rng.random_vec(64));
2✔
127
         const Botan::Roughtime::Nonce nonce_v(rand64);
1✔
128
         result.test_is_true("nonce from vector",
1✔
129
                             nonce_v.get_nonce() == Botan::typecast_copy<std::array<uint8_t, 64>>(rand64.data()));
1✔
130
         const Botan::Roughtime::Nonce nonce_a(Botan::typecast_copy<std::array<uint8_t, 64>>(rand64.data()));
1✔
131
         result.test_is_true("nonce from array",
1✔
132
                             nonce_v.get_nonce() == Botan::typecast_copy<std::array<uint8_t, 64>>(rand64.data()));
1✔
133
         rand64.push_back(10);
1✔
134
         result.test_throws("vector oversize",
1✔
135
                            [&rand64]() { const Botan::Roughtime::Nonce nonce_v2(rand64); });  //size 65
2✔
136
         rand64.pop_back();
1✔
137
         rand64.pop_back();
1✔
138
         result.test_throws("vector undersize",
1✔
139
                            [&rand64]() { const Botan::Roughtime::Nonce nonce_v2(rand64); });  //size 63
2✔
140

141
         return result;
1✔
142
      }
1✔
143

144
      static Test::Result test_chain(Botan::RandomNumberGenerator& rng) {
1✔
145
         Test::Result result("roughtime chain");
1✔
146

147
         Botan::Roughtime::Chain c1;
1✔
148
         result.test_is_true("default constructed is empty", c1.links().empty() && c1.responses().empty());
1✔
149

150
         auto rand64 = Botan::unlock(rng.random_vec(64));
2✔
151
         const Botan::Roughtime::Nonce nonce_v(rand64);
1✔
152
         result.test_is_true(
2✔
153
            "empty chain nonce is blind",
154
            c1.next_nonce(nonce_v).get_nonce() == Botan::typecast_copy<std::array<uint8_t, 64>>(rand64.data()));
1✔
155

156
         const std::string chain_str =
1✔
157
            "ed25519 bbT+RPS7zKX6w71ssPibzmwWqU9ffRV5oj2OresSmhE= eu9yhsJfVfguVSqGZdE8WKIxaBBM0ZG3Vmuc+IyZmG2YVmrIktUByDdwIFw6F4rZqmSFsBO85ljoVPz5bVPCOw== BQAAAEAAAABAAAAApAAAADwBAABTSUcAUEFUSFNSRVBDRVJUSU5EWBnGOEajOwPA6G7oL47seBP4C7eEpr57H43C2/fK/kMA0UGZVUdf4KNX8oxOK6JIcsbVk8qhghTwA70qtwpYmQkDAAAABAAAAAwAAABSQURJTUlEUFJPT1RAQg8AJrA8tEqPBQAqisiuAxgy2Pj7UJAiWbCdzGz1xcCnja3T+AqhC8fwpeIwW4GPy/vEb/awXW2DgSLKJfzWIAz+2lsR7t4UjNPvAgAAAEAAAABTSUcAREVMRes9Ch4X0HIw5KdOTB8xK4VDFSJBD/G9t7Et/CU7UW61OiTBXYYQTG2JekWZmGa0OHX1JPGG+APkpbsNw0BKUgYDAAAAIAAAACgAAABQVUJLTUlOVE1BWFR/9BWjpsWTQ1f6iUJea3EfZ1MkX3ftJiV3ABqNLpncFwAAAAAAAAAA//////////8AAAAA\n"
158
            "ed25519 gD63hSj3ScS+wuOeGrubXlq35N1c5Lby/S+T7MNTjxo= uLeTON9D+2HqJMzK6sYWLNDEdtBl9t/9yw1cVAOm0/sONH5Oqdq9dVPkC9syjuWbglCiCPVF+FbOtcxCkrgMmA== BQAAAEAAAABAAAAApAAAADwBAABTSUcAUEFUSFNSRVBDRVJUSU5EWOw1jl0uSiBEH9HE8/6r7zxoSc01f48vw+UzH8+VJoPelnvVJBj4lnH8uRLh5Aw0i4Du7XM1dp2u0r/I5PzhMQoDAAAABAAAAAwAAABSQURJTUlEUFJPT1RAQg8AUBo+tEqPBQC47l77to7ESFTVhlw1SC74P5ssx6gpuJ6eP+1916GuUiySGE/x3Fp0c3otUGAdsRQou5p9PDTeane/YEeVq4/8AgAAAEAAAABTSUcAREVMRe5T1ml8wHyWAcEtHP/U5Rg/jFXTEXOSglngSa4aI/CECVdy4ZNWeP6vv+2//ZW7lQsrWo7ZkXpvm9BdBONRSQIDAAAAIAAAACgAAABQVUJLTUlOVE1BWFQpXlenV0OfVisvp9jDHXLw8vymZVK9Pgw9k6Edf8ZEhUgSGEc5jwUASHLvZE2PBQAAAAAA\n";
1✔
159

160
         Botan::Roughtime::Chain c2(chain_str);
1✔
161
         result.test_is_true("have two elements", c2.links().size() == 2 && c2.responses().size() == 2);
2✔
162
         result.test_is_true("serialize loopback", c2.to_string() == chain_str);
1✔
163

164
         c1.append(c2.links()[0], 1);
1✔
165
         result.test_is_true("append ok", c1.links().size() == 1 && c1.responses().size() == 1);
2✔
166
         c1.append(c2.links()[1], 1);
1✔
167
         result.test_is_true("max size", c1.links().size() == 1 && c1.responses().size() == 1);
2✔
168

169
         result.test_throws("non-positive max chain size", [&]() { c1.append(c2.links()[1], 0); });
2✔
170
         result.test_throws("1 field", [&]() { const Botan::Roughtime::Chain a("ed25519"); });
2✔
171
         result.test_throws("2 fields", [&]() {
1✔
172
            const Botan::Roughtime::Chain a("ed25519 bbT+RPS7zKX6w71ssPibzmwWqU9ffRV5oj2OresSmhE=");
1✔
173
         });
×
174
         result.test_throws("3 fields", [&]() {
1✔
175
            const Botan::Roughtime::Chain a(
1✔
176
               "ed25519 bbT+RPS7zKX6w71ssPibzmwWqU9ffRV5oj2OresSmhE= eu9yhsJfVfguVSqGZdE8WKIxaBBM0ZG3Vmuc+IyZmG2YVmrIktUByDdwIFw6F4rZqmSFsBO85ljoVPz5bVPCOw==");
1✔
177
         });
×
178
         result.test_throws("5 fields", [&]() {
1✔
179
            const Botan::Roughtime::Chain a(
1✔
180
               "ed25519 bbT+RPS7zKX6w71ssPibzmwWqU9ffRV5oj2OresSmhE= eu9yhsJfVfguVSqGZdE8WKIxaBBM0ZG3Vmuc+IyZmG2YVmrIktUByDdwIFw6F4rZqmSFsBO85ljoVPz5bVPCOw== BQAAAEAAAABAAAAApAAAADwBAABTSUcAUEFUSFNSRVBDRVJUSU5EWBnGOEajOwPA6G7oL47seBP4C7eEpr57H43C2/fK/kMA0UGZVUdf4KNX8oxOK6JIcsbVk8qhghTwA70qtwpYmQkDAAAABAAAAAwAAABSQURJTUlEUFJPT1RAQg8AJrA8tEqPBQAqisiuAxgy2Pj7UJAiWbCdzGz1xcCnja3T+AqhC8fwpeIwW4GPy/vEb/awXW2DgSLKJfzWIAz+2lsR7t4UjNPvAgAAAEAAAABTSUcAREVMRes9Ch4X0HIw5KdOTB8xK4VDFSJBD/G9t7Et/CU7UW61OiTBXYYQTG2JekWZmGa0OHX1JPGG+APkpbsNw0BKUgYDAAAAIAAAACgAAABQVUJLTUlOVE1BWFR/9BWjpsWTQ1f6iUJea3EfZ1MkX3ftJiV3ABqNLpncFwAAAAAAAAAA//////////8AAAAA abc");
1✔
181
         });
×
182
         result.test_throws("invalid key type", [&]() {
1✔
183
            const Botan::Roughtime::Chain a(
1✔
184
               "rsa bbT+RPS7zKX6w71ssPibzmwWqU9ffRV5oj2OresSmhE= eu9yhsJfVfguVSqGZdE8WKIxaBBM0ZG3Vmuc+IyZmG2YVmrIktUByDdwIFw6F4rZqmSFsBO85ljoVPz5bVPCOw== BQAAAEAAAABAAAAApAAAADwBAABTSUcAUEFUSFNSRVBDRVJUSU5EWBnGOEajOwPA6G7oL47seBP4C7eEpr57H43C2/fK/kMA0UGZVUdf4KNX8oxOK6JIcsbVk8qhghTwA70qtwpYmQkDAAAABAAAAAwAAABSQURJTUlEUFJPT1RAQg8AJrA8tEqPBQAqisiuAxgy2Pj7UJAiWbCdzGz1xcCnja3T+AqhC8fwpeIwW4GPy/vEb/awXW2DgSLKJfzWIAz+2lsR7t4UjNPvAgAAAEAAAABTSUcAREVMRes9Ch4X0HIw5KdOTB8xK4VDFSJBD/G9t7Et/CU7UW61OiTBXYYQTG2JekWZmGa0OHX1JPGG+APkpbsNw0BKUgYDAAAAIAAAACgAAABQVUJLTUlOVE1BWFR/9BWjpsWTQ1f6iUJea3EfZ1MkX3ftJiV3ABqNLpncFwAAAAAAAAAA//////////8AAAAA");
1✔
185
         });
×
186
         result.test_throws("invalid key", [&]() {
1✔
187
            const Botan::Roughtime::Chain a(
1✔
188
               "ed25519 bbT+RPS7zKX6wssPibzmwWqU9ffRV5oj2OresSmhE= eu9yhsJfVfguVSqGZdE8WKIxaBBM0ZG3Vmuc+IyZmG2YVmrIktUByDdwIFw6F4rZqmSFsBO85ljoVPz5bVPCOw== BQAAAEAAAABAAAAApAAAADwBAABTSUcAUEFUSFNSRVBDRVJUSU5EWBnGOEajOwPA6G7oL47seBP4C7eEpr57H43C2/fK/kMA0UGZVUdf4KNX8oxOK6JIcsbVk8qhghTwA70qtwpYmQkDAAAABAAAAAwAAABSQURJTUlEUFJPT1RAQg8AJrA8tEqPBQAqisiuAxgy2Pj7UJAiWbCdzGz1xcCnja3T+AqhC8fwpeIwW4GPy/vEb/awXW2DgSLKJfzWIAz+2lsR7t4UjNPvAgAAAEAAAABTSUcAREVMRes9Ch4X0HIw5KdOTB8xK4VDFSJBD/G9t7Et/CU7UW61OiTBXYYQTG2JekWZmGa0OHX1JPGG+APkpbsNw0BKUgYDAAAAIAAAACgAAABQVUJLTUlOVE1BWFR/9BWjpsWTQ1f6iUJea3EfZ1MkX3ftJiV3ABqNLpncFwAAAAAAAAAA//////////8AAAAA");
1✔
189
         });
×
190
         result.test_throws("invalid nonce", [&]() {
1✔
191
            const Botan::Roughtime::Chain a(
1✔
192
               "ed25519 bbT+RPS7zKX6w71ssPibzmwWqU9ffRV5oj2OresSmhE= eu9yhsJfVfguVSqGZdE8WKIxaBBM0ZG3Vmuc+IyZmG2UByDdwIFw6F4rZqmSFsBO85ljoVPz5bVPCOw== BQAAAEAAAABAAAAApAAAADwBAABTSUcAUEFUSFNSRVBDRVJUSU5EWBnGOEajOwPA6G7oL47seBP4C7eEpr57H43C2/fK/kMA0UGZVUdf4KNX8oxOK6JIcsbVk8qhghTwA70qtwpYmQkDAAAABAAAAAwAAABSQURJTUlEUFJPT1RAQg8AJrA8tEqPBQAqisiuAxgy2Pj7UJAiWbCdzGz1xcCnja3T+AqhC8fwpeIwW4GPy/vEb/awXW2DgSLKJfzWIAz+2lsR7t4UjNPvAgAAAEAAAABTSUcAREVMRes9Ch4X0HIw5KdOTB8xK4VDFSJBD/G9t7Et/CU7UW61OiTBXYYQTG2JekWZmGa0OHX1JPGG+APkpbsNw0BKUgYDAAAAIAAAACgAAABQVUJLTUlOVE1BWFR/9BWjpsWTQ1f6iUJea3EfZ1MkX3ftJiV3ABqNLpncFwAAAAAAAAAA//////////8AAAAA");
1✔
193
         });
×
194

195
         return result;
1✔
196
      }
2✔
197

198
      static Test::Result test_server_information() {
1✔
199
         Test::Result result("roughtime server_information");
1✔
200

201
         const auto servers = Botan::Roughtime::servers_from_str(
1✔
202
            "Chainpoint-Roughtime ed25519 bbT+RPS7zKX6w71ssPibzmwWqU9ffRV5oj2OresSmhE= udp roughtime.chainpoint.org:2002\n"
203
            "Cloudflare-Roughtime ed25519 0GD7c3yP8xEc4Zl2zeuN2SlLvDVVocjsPSL8/Rl/7zg= udp roughtime.cloudflare.com:2003\n"
204
            "Google-Sandbox-Roughtime ed25519 etPaaIxcBMY1oUeGpwvPMCJMwlRVNxv51KK/tktoJTQ= udp roughtime.sandbox.google.com:2002\n"
205
            "int08h-Roughtime ed25519 AW5uAoTSTDfG5NfY1bTh08GUnOqlRb+HVhbJ3ODJvsE= udp roughtime.int08h.com:2002\n"
206
            "ticktock ed25519 cj8GsiNlRkqiDElAeNMSBBMwrAl15hYPgX50+GWX/lA= udp ticktock.mixmin.net:5333\n");
1✔
207

208
         result.test_is_true("size", servers.size() == 5);
1✔
209
         result.test_str_eq("name", servers[0].name(), "Chainpoint-Roughtime");
1✔
210
         result.test_str_eq("name", servers[4].name(), "ticktock");
1✔
211
         result.test_is_true(
2✔
212
            "public key",
213
            servers[0].public_key().get_public_key() ==
1✔
214
               Botan::Ed25519_PublicKey(Botan::base64_decode("bbT+RPS7zKX6w71ssPibzmwWqU9ffRV5oj2OresSmhE="))
1✔
215
                  .get_public_key());
216
         result.test_is_true("single address", servers[0].addresses().size() == 1);
1✔
217
         result.test_str_eq("address", servers[0].addresses()[0], "roughtime.chainpoint.org:2002");
1✔
218

219
         result.test_throws("1 field", [&]() { Botan::Roughtime::servers_from_str("A"); });
2✔
220
         result.test_throws("2 fields", [&]() { Botan::Roughtime::servers_from_str("A ed25519"); });
2✔
221
         result.test_throws("3 fields", [&]() {
1✔
222
            Botan::Roughtime::servers_from_str("A ed25519 bbT+RPS7zKX6w71ssPibzmwWqU9ffRV5oj2OresSmhE=");
1✔
223
         });
×
224
         result.test_throws("4 fields", [&]() {
1✔
225
            Botan::Roughtime::servers_from_str("A ed25519 bbT+RPS7zKX6w71ssPibzmwWqU9ffRV5oj2OresSmhE= udp");
1✔
226
         });
×
227
         result.test_throws("invalid address", [&]() {
1✔
228
            Botan::Roughtime::servers_from_str("A ed25519 bbT+RPS7zKX6w71ssPibzmwWqU9ffRV5oj2OresSmhE= udp ");
1✔
229
         });
×
230
         result.test_throws("invalid key type", [&]() {
1✔
231
            Botan::Roughtime::servers_from_str(
1✔
232
               "A rsa bbT+RPS7zKX6w71ssPibzmwWqU9ffRV5oj2OresSmhE= udp roughtime.chainpoint.org:2002");
233
         });
×
234
         result.test_throws("invalid key", [&]() {
1✔
235
            Botan::Roughtime::servers_from_str(
1✔
236
               "A ed25519 bbT+RP7zKX6w71ssPibzmwWqU9ffRV5oj2OresSmhE= udp roughtime.chainpoint.org:2002");
237
         });
×
238
         result.test_throws("invalid protocol", [&]() {
1✔
239
            Botan::Roughtime::servers_from_str(
1✔
240
               "A ed25519 bbT+RPS7zKX6w71ssPibzmwWqU9ffRV5oj2OresSmhE= tcp roughtime.chainpoint.org:2002");
241
         });
×
242

243
         return result;
1✔
244
      }
1✔
245

246
      static Test::Result test_request_online(Botan::RandomNumberGenerator& rng) {
1✔
247
         Test::Result result("roughtime request online");
1✔
248

249
         const Botan::Roughtime::Nonce nonce(rng);
1✔
250
         try {
1✔
251
            const auto response_raw =
1✔
252
               Botan::Roughtime::online_request("roughtime.cloudflare.com:2003", nonce, std::chrono::seconds(5));
1✔
253
            const auto now = std::chrono::system_clock::now();
1✔
254
            const auto response = Botan::Roughtime::Response::from_bits(response_raw, nonce);
1✔
255
            const std::chrono::milliseconds local_clock_max_error(1000);
1✔
256
            const auto diff_abs =
1✔
257
               now >= response.utc_midpoint() ? now - response.utc_midpoint() : response.utc_midpoint() - now;
1✔
258
            result.test_is_true("online", diff_abs <= (response.utc_radius() + local_clock_max_error));
1✔
259
         } catch(const std::exception& e) {
1✔
260
            result.test_failure(e.what());
×
261
         }
×
262
         return result;
1✔
263
      }
×
264

265
   public:
266
      std::vector<Test::Result> run() override {
1✔
267
         auto& rng = this->rng();
1✔
268

269
         std::vector<Test::Result> results{test_nonce(rng), test_chain(rng), test_server_information()};
4✔
270

271
         if(Test::options().run_online_tests()) {
1✔
272
            results.push_back(test_request_online(rng));
2✔
273
         }
274

275
         return results;
1✔
276
      }
1✔
277
};
278

279
BOTAN_REGISTER_TEST("roughtime", "roughtime_tests", Roughtime);
280

281
#endif
282

283
}  // namespace
284

285
}  // namespace Botan_Tests
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