rippled
Loading...
Searching...
No Matches
ValidatorSite_test.cpp
1#include <test/jtx.h>
2#include <test/jtx/TrustedPublisherServer.h>
3#include <test/unit_test/FileDirGuard.h>
4
5#include <xrpld/app/misc/ValidatorSite.h>
6
7#include <xrpl/basics/strHex.h>
8#include <xrpl/protocol/PublicKey.h>
9#include <xrpl/protocol/jss.h>
10
11#include <boost/algorithm/string/join.hpp>
12#include <boost/algorithm/string/predicate.hpp>
13#include <boost/asio.hpp>
14#include <boost/range/adaptor/transformed.hpp>
15
16#include <date/date.h>
17
18#include <chrono>
19
20namespace ripple {
21namespace detail {
22constexpr char const*
24{
25 return R"vl({
26 "public_key": "ED2677ABFFD1B33AC6FBC3062B71F1E8397C1505E1C42C64D11AD1B28FF73F4734",
27 "manifest": "JAAAAAFxIe0md6v/0bM6xvvDBitx8eg5fBUF4cQsZNEa0bKP9z9HNHMh7V0AnEi5D4odY9X2sx+cY8B3OHNjJvMhARRPtTHmWnAhdkDFcg53dAQS1WDMQDLIs2wwwHpScrUnjp1iZwwTXVXXsaRxLztycioto3JgImGdukXubbrjeqCNU02f7Y/+6w0BcBJA3M0EOU+39hmB8vwfgernXZIDQ1+o0dnuXjX73oDLgsacwXzLBVOdBpSAsJwYD+nW8YaSacOHEsWaPlof05EsAg==",
28 "blob" : "{"sequence":37,"expiration":594172800,"validators":[{"validation_public_key":"ED49F8EB8071E26D4E067238D62D67CA6EBFB298B4A72DCFAD536A7EC8925C519B","manifest":"JAAAAAFxIe1J+OuAceJtTgZyONYtZ8puv7KYtKctz61Tan7IklxRm3MhAogMmQTRAFK/+m/XWQfJvXTeMcaiyF6gkhgAssdLOciIdkYwRAIgNxYavJPyHANuxogQY7RNBP9tZogZJxF1wZqzm4a37GgCIBOvbeQ8+P46CCIcy4ZCZA5rZZ+93H0aKJZQ3+IA0OOucBJAASTpQKtEJkeqwHz+jKWNvWl0sHRQZSSJuYiMyyUa7ytpsev6NGoFn1bNpYTgvsaPxoz7vHskDh2Z6gebSDCRCA=="},{"validation_public_key":"ED8108F7D9EE779DC1075EC3CF8EB9004DDF84CD0A0FF5775C9E375CCACDAA3022","manifest":"JAAAAAFxIe2BCPfZ7nedwQdew8+OuQBN34TNCg/1d1yeN1zKzaowInMhAlvPJANQcRUSNUR+NlZuo4kGpYF8l9pHPP1ZROOUcNYRdkYwRAIgRWZZuJLAf3SNMGoFygUlu+eAD51ZoHFMcyoicn0WKucCIGw4n9cVDPOBGC8wlJDy9282x9RdOwo1co2g+NZlc/DNcBJAKJhK/am6L7KE450NVzhpVVoL8OFMo0mFxk8BDG6QO2cSNMN3Oywf40D+Iles9LXxxvvPB6gWSmPlwF7dNR9GAQ=="},{"validation_public_key":"ED7E3BAFF901DE525B00B2C1FE19AF449A080B5F3100C6EC182C867AF61F710FFD","manifest":"JAAAAAFxIe1+O6/5Ad5SWwCywf4Zr0SaCAtfMQDG7Bgshnr2H3EP/XMhA5+GDJh2ApWKCh2N5omx4FAOFEqXMjqy1HJ5E7WA3FrOdkcwRQIhAMRezwjTAL35jpjj15jqoHUvRm7+7iHpU47aKE6HEChZAiBSMIZaMWv6qNJHniYpsYHx4OPPpBoCMMdMVAGfFi9fK3ASQE0TYiIsGv0/ylpqGEBiLkk2XjrA8++Akzpr9vcTtrkHiDDo0cHKO9mRULAXesRrOyFb5QcO0l0VswevD5jlswk="},{"validation_public_key":"ED45E80A04D79CB9DF00AEBD86DCDC1686D6419EA9E5E0E71F1A817E08B5076A55","manifest":"JAAAAAFxIe1F6AoE15y53wCuvYbc3BaG1kGeqeXg5x8agX4ItQdqVXMhAxZo157pcB9de6Smk7hoK3wNCAr4aFZtfAPi7CE4mNJldkcwRQIhALlVjXCfiy/mtXBWsNt77t4jKcNEBpRV8zv+SpU5lCh0AiBa8vo8xxpviYlf4zdG+nQhB2OgfkQZZPMHOt7CaXzXgXASQL8O5p083mg4KKL8uZfMaUqdgzuJ0Gta1lyUWPctTPCxY135XwK+nJAdFsIUFNJ9MPjnpCmSjYVzVa6M5/nAcAI="},{"validation_public_key":"EDD8C88642795CE69C5B780E01702C370F9507D0B64433F17EFE70F2637A40ADB7","manifest":"JAAAAAFxIe3YyIZCeVzmnFt4DgFwLDcPlQfQtkQz8X7+cPJjekCtt3MhAnFfr+r9BXdsXE/cBlJMyd/XsO1A5XEYCctrsvLEX+DmdkcwRQIhANRcRMg9SAXoaOvHDZ2av9RzEaZaVENfQiVgsi+Ox3F0AiB2snSIOm6c4/inbtU0UmWLQTzuwkOdUFPIB8Ax8dmGuHASQMUIfXMj96kcFTSJnMFC/mW/AQ8bKXkFrrk0CUTFFKweEjTq+STrFi6qLL2MT7nveGxsXBCgztjc0qGas9KFWgM="},{"validation_public_key":"EDBDEB901F7C75D0E20C6C42AF03BE0DA40377AF1939A18B3CB3679661DD5F9F74","manifest":"JAAAAAFxIe2965AffHXQ4gxsQq8Dvg2kA3evGTmhizyzZ5Zh3V+fdHMhAg3cyKNPMPqKgR7kIi7c/8GL/YgdBtg4mSAWvwmaevVGdkYwRAIgWzG8GqYg3YpwDs8xXa9XqLHss76KT2uAHRhUXFVUqCQCIG2EvbFKnxezRd9cpPHSt32HXK+P4+aL3p2+vqlCxRR9cBJAboXTmYTayocA3zf9dWEXtyaeOGC1k5WdYURzPleevvalR4xVoXzs38iGPxFr/pA9nL+M4duu0GKCHlVir+fBAg=="},{"validation_public_key":"EDA17871E72B0C570AC4345C60CF02AFBBB740A631B7AD0E1E573216574D9AEA02","manifest":"JAAAAAFxIe2heHHnKwxXCsQ0XGDPAq+7t0CmMbetDh5XMhZXTZrqAnMhAojyuzgtreQkxQj8prHxOsbDcF5fu4XXb0KxEL/Pq5HhdkcwRQIhANfPDLZP47aCWwt5kBnp75BuuCgp9c4BfJPd66SFCw61AiAJvegBvvPIrec+XOSzKRfi5uuXWxtl9Eyr2aPBYXvbRHASQMULYEo7beRfoUCnjk1sTYyY91tLIGLgnnaWXhUm80+zs5IGegk8qijKAtBOMuBC71lAB4KhJc+dB2rpMOFc5gw="},{"validation_public_key":"EDF46EE27AD0E1A714AFECDA816EAB7114614FCB92D0CB4D97B6A88ED43434AFC9","manifest":"JAAAAAFxIe30buJ60OGnFK/s2oFuq3EUYU/LktDLTZe2qI7UNDSvyXMhAw0ATWjVTt4FfeKO7kv6fFgd/go2+d5BSyUcURmRWnTtdkcwRQIhAMwOgDec7QYYNngspg90wEvVbsoh2ux14RPTw+GHaXNlAiALgfEsz+AF4eyX/Y5i44VrFjFFIMWUfOZaQJtsxteM1XASQLOaF0t2ZpqVKd8JESQVY+zU567iAAG2amTPZx95875S9A6Pl+kH5TGHMAeWjgWSqfh3m2HBJX7NIcXb98vy9AA="},{"validation_public_key":"ED6E4C41E59FFBEB51726E54468502FE6437238FA78EA51634E7BF0D09171AEE8F","manifest":"JAAAAAFxIe1uTEHln/vrUXJuVEaFAv5kNyOPp46lFjTnvw0JFxruj3MhAuztGWb/Oi1/V5m5dujWr9HmbKRyK4XYk+kmuFPSgAFrdkYwRAIgfQ+BgXX6QblZy4H05o7GPSIwqS7QQRUW7dqF54IAiiMCIH4XfLw956iEaoxZOk7Kctin2X9hMfaLN7wys9yAUFoZcBJAueEi84XR3Ll1GLJWanW1g1MdUj/0PAxJbw6EEQRuG3zdnuRHNXld6UZAbIkVcP0ztfqulBzjbcsLDOKFEicSBg=="},{"validation_public_key":"EDB6FC8E803EE8EDC2793F1EC917B2EE41D35255618DEB91D3F9B1FC89B75D4539","manifest":"JAAAAAFxIe22/I6APujtwnk/HskXsu5B01JVYY3rkdP5sfyJt11FOXMhA8VdvHFyScByQGTYNGeOvB0+67gWaqefcfvRk5+KwgV1dkYwRAIgZFulO/AiMoczng6i/4BkfzT7j9lxF4PP1ufgrOQaJ8sCIBX/E8Zbpn7tWqgAyNyWpVPkhFmaUMqEry8WoUT1fdGQcBJAv51RqJxgg/VrnrZwiLK2Dc0CKbiLPO5HJ4ZMsjdPT2gRc97rWkAXuV2L6PNFO59xyuoaZmSMlZYvqSGPpfF7Bw=="},{"validation_public_key":"ED691303992FEC64E6BC4BACD36AE6E5AEDC23F2861B6D8EFB9FD77EE3EADE3435","manifest":"JAAAAAFxIe1pEwOZL+xk5rxLrNNq5uWu3CPyhhttjvuf137j6t40NXMhAi2AXJQgo/JuW3r7f/6CcVsGN1YmIj11GiIESHBnQSk8dkcwRQIhANCDEQymrd6veT3ouacF6fhBr5wLw3GmXg1rMCLVvBzZAiA8uWQ+tqd46WmfBexjSBQ2Jd6UAGdrHvjcCQ2ZgSooCnASQFkHl+D7/U3WByYP384+pcFDf2Gi4WIRHVTo58cqdk5CDiwc1T0rDoLhmo41a3f+dsftfwR4aMmwFcPXLnrjrAI="},{"validation_public_key":"EDAD16667F0185DDBB7FA65B22F4B7D310BF5C3E2D9B823FB06A3A41AF8AC83BC1","manifest":"JAAAAAFxIe2tFmZ/AYXdu3+mWyL0t9MQv1w+LZuCP7BqOkGvisg7wXMhAqweE3PIS3E44KhMqKjKtbkBe8H8GbiuoAXAYDRoVRHodkYwRAIgagGkXtowUybdltKojv0lvvflrlQ9IRnPOjekF60iHzgCICg6ZocIMzkUuvO91BEormIWmX4G/MGT2zro6I/PvB8XcBJAcJLXkt/w/kcwEvNiZmi2i2nMn1wiP3LS9NJjBPju8KFLAMg0O9ydQT67U/ALYOeTPTO2/i2Yw9OSlibtqhgzDA=="},{"validation_public_key":"EDC245027A52EE5318095598EC3AB65FF4A3B9F9428E10B2F3C6F39DE15A15C90A","manifest":"JAAAAAFxIe3CRQJ6Uu5TGAlVmOw6tl/0o7n5Qo4QsvPG853hWhXJCnMhA/8/9rKUdA61j/fIEP/cqLpxBlmIhP2rg1d7NaEPyKV+dkcwRQIhAIxE0M/FJ50vfZW6fPpy4yCZumY9n0obrOojUkjm55a0AiBj56O0MpopGoY9HxC/+4wNO36Ho7E9CQeHsnKreDdsAXASQIYUd81jbiVUlET4dGoG2p+cf+2GqEXX5fJMSSyX/qe0XfR4cO+4qlgmjMQdCRDBWABHVvdN/yZyi/rL2c+WrQc="},{"validation_public_key":"ED4246AA3AE9D29863944800CCA91829E4447498A20CD9C3973A6B59346C75AB95","manifest":"JAAAAAFxIe1CRqo66dKYY5RIAMypGCnkRHSYogzZw5c6a1k0bHWrlXMhAkm1lz0c8QXWfJ9b1vB72dLabw8wYId8MtnpsHHBEC8pdkYwRAIgQlb6HJ53hsTAfVid+AOdBVvMF7rahIKNLBHUgn52zBECIGLUqFu8a1AAHRJcVonKYEnmhJwbCXLn+je7na1WD1/ocBJAE4vfvrGSmZC2uAUGmM5dIBtoSgEUey+2VleDYEsce94txYcjR8Z7QLNaliD8w/bD5/hvYQ8meV1Wg1jJFNe0CA=="},{"validation_public_key":"ED2C1468B4A11D281F93EF337C95E4A08DF0000FDEFB6D0EA9BC05FBD5D61A1F5A","manifest":"JAAAAAFxIe0sFGi0oR0oH5PvM3yV5KCN8AAP3vttDqm8BfvV1hofWnMhAkMUmCD2aPmgFDDRmimvSicSIScw6YNr42Dw4RAdwrOAdkcwRQIhAJFOHMg6qTG8v60dhrenYYk6cwOaRXq0RNmLjyyCiz5lAiAdU0YkDUJQhnN8Ry8s+6zTJLiNLbtM8oO/cLnurVpRM3ASQGALarHAsJkSZQtGdM2AaR/joFK/jhDU57+l+RSYjri/ydE20DaKanwkMEoVlBTg7lX4hYjEnmkqo73wIthLOAQ="},{"validation_public_key":"EDA54C85F91219FD259134B6B126AD64AE7204B81DD4052510657E1A5697246AD2","manifest":"JAAAAAJxIe2lTIX5Ehn9JZE0trEmrWSucgS4HdQFJRBlfhpWlyRq0nMhAl8cJerPv+vo1BK611xVTpGxjjr/CuxPTgU8URM4eTZ5dkYwRAIgdK3cQV2Y/viZne/PboKSKewngTuIN2M6c8azwqc20uUCIAc6GoNT+P2YBy49gdau4P7ySwWoQX5nf9dQxiQav5wIcBJAqiCK0d6QRZSpiVHp8O9nlKXCSEhsiSNcWcEFm/fGhJAnAN0Ov9HINId1pxrBn2dKRegLTvYG3Bpbz//HLgEdDA=="},{"validation_public_key":"ED9AE4F5887BA029EB7C0884486D23CF281975F773F44BD213054219882C411CC7","manifest":"JAAAAAFxIe2a5PWIe6Ap63wIhEhtI88oGXX3c/RL0hMFQhmILEEcx3MhAmG2zgv8FBZsZJU8aPapwo9cIqQv4/MSS1oVA5eVMiwLdkYwRAIgF+LOe4eY0gp9ttqh2gnv+z75OqLyOQMpGPALgm+NtOsCICDXBZVPtprmBDkBJkPFSnE55D9eKYRH8z/iY1EtpNplcBJAADEWGVT80Owhd1lh2JsU/oZlmeNF5WN7YvlB8llExaRKEVC+GW9Wg+iNIQ3rmV7P8aNaVuaabG00fOgkgzNhDw=="},{"validation_public_key":"EDA8D29F40CEB28995617641A3BC42692E1DE883214F612FBB62087A148E5F6F9A","manifest":"JAAAAAFxIe2o0p9AzrKJlWF2QaO8QmkuHeiDIU9hL7tiCHoUjl9vmnMhAnYnP7Eg6VgNnEUTRE29d64jQT/iBcWTQtNrUzyD6MJ+dkcwRQIhAOEsV5anTkloSmTZRbimMyBKqHoJYXcBBe8lLiPYC7mUAiAz2aNOpfQ/1LycWloIMvdhxzinq5X7Uas/uOSb9wh8d3ASQLVkfpW/GO6wdT6AuuSJ56TtM343pDNH+iSzxltIfdrPiUxT5rf4k21lQQuPClXm9+SfKrCiUXZK7dj0/GWTYQg="},{"validation_public_key":"ED38B0288EA240B4CDEC18A1A6289EB49007E4EBC0DE944803EB7EF141C5664073","manifest":"JAAAAAFxIe04sCiOokC0zewYoaYonrSQB+TrwN6USAPrfvFBxWZAc3MhAgOKcvIuchalrZw/glTuOxV3IOCcporxMB7JqAVupk1edkcwRQIhAOvRzpe+IYZK1MyInIQZ87JvP2J8SIXCXZMPBCdITBamAiASavJXi9pws8rDDJSxhGMlmE7zI5bSA8ivtRC9Lgq+UXASQDl3eoqLID+ETJNM+zbMuvwvcHEIxeBZkZ9fp5jJv6OCTPwlj4TJSuy1avEWqUYS2riv5Dvl2haFUoCHf4yawAA="},{"validation_public_key":"EDEE10DC36ACD995C8E0E86E3CD2FBF8301A4AC2B8847B61A1935DE4973B407C0E","manifest":"JAAAAAFxIe3uENw2rNmVyODobjzS+/gwGkrCuIR7YaGTXeSXO0B8DnMhAmX0vb7j+lgBjFjbN9RlA86J7AO2Vn6HLquO3aisK4mwdkYwRAIgfxBLn7i4jg/di0U25q6kIbVfTzqbA0SCpQ0I57TOFkcCIFMtJQpENjB2K2EmvBHPvNcwuSPc3vsEeqE2rNJ/cT5DcBJAf68XPFu5RjCeLgpFJM7PKFLgoV8e1nxO5ewjq9Q+TAEGnFyS0IOwf6pOOtIVMdVeXu1v6p4fhXQkdihHt1x6Ag=="},{"validation_public_key":"ED583ECD06C3B7369980E65C78C440A529300F557ED81256283F7DD5AA3513A334","manifest":"JAAAAAFxIe1YPs0Gw7c2mYDmXHjEQKUpMA9VftgSVig/fdWqNROjNHMhAyuUnzZZ1n2/GaTmE1m7H/v9YlZyDEwHY3gSHUA3ICL9dkYwRAIgHx2PHvidoN+5yG9WeAS2k7nwIM8ajxQW6wjvt8kBenACIDNxQPQkDyDJH9seS5C62mAarQmgiN89YS3jhNtnvEIqcBJAj7Jh0Kac+aJdpoepu/+eJKnnFQ7YByZB8eMZ+SS1zLhE+lip/49qqVNcpAxEqfaGtxJzoDDD1/QbuU7NOSPkCg=="},{"validation_public_key":"ED95C5172B2AD7D39434EEBC436B65B3BB7E58D5C1CEFC820B6972ACAD776E286A","manifest":"JAAAAAFxIe2VxRcrKtfTlDTuvENrZbO7fljVwc78ggtpcqytd24oanMhAiqcRde3MQZ075fa4ZNNyRaYJGMdBNkBnn3bQrKseBDQdkYwRAIgU+LfcE71DPVrO+KtUBjQ9D2u0k/Pr7lukO1nPRj6hSACIDNLYC/JFgobCsIa0BGw+6bUnOw9meU3FdXgR7Q7SoqJcBJAXQakOoQnPp3pcLL7zdKCPUX4b+/FC9Unhqp+O9xQFnRaCWVGmk5MJOIMs4WOQdpM1j3OgSsABmRuCXYvwo/nDw=="},{"validation_public_key":"ED90163D2BF0B7788904C4A4118D7D968920E847D88B79178390837DE3CA261562","manifest":"JAAAAAFxIe2QFj0r8Ld4iQTEpBGNfZaJIOhH2It5F4OQg33jyiYVYnMhA72VTRiGhkJBtqgGHDzHj7YbC6+NsEKrFHNuE/LO3Tn5dkYwRAIgf8s+fYt0llrKQ2qiWPnGmb6qJPoe8OnCM3VS29XKbYYCIHGnlJ4OTs2dXugO6Bto63NpDvvqJ+WIwdYKqZ6BiBfzcBJAGvNtkog4pfE5dZRwmic87ZBeeunOh4YpL0SERdxWj43Cs9815zFJuZysSaUX2R/vdE2VKqvSgqqtDEnrMo2oAw=="}]}",
29 "signature" : "9FF30EDC4DED7ABCD0D36389B7C716EED4B5E4F043902853534EBAC7BE966BB3813D5CF25E4DADA5E657CCF019FFD11847FD3CC44B5559A6FCEEE4C3DCFF8D0E",
30 "version": 1
31}
32)vl";
33}
34
37} // namespace detail
38
39namespace test {
41{
42private:
44
45 void
47 {
48 testcase("Config Load");
49
50 using namespace jtx;
51
52 Env env(*this, envconfig(), nullptr, beast::severities::kDisabled);
53 auto trustedSites =
55
56 // load should accept empty sites list
57 std::vector<std::string> emptyCfgSites;
58 BEAST_EXPECT(trustedSites->load(emptyCfgSites));
59
60 // load should accept valid validator site uris
62 {"http://ripple.com/",
63 "http://ripple.com/validators",
64 "http://ripple.com:8080/validators",
65 "http://207.261.33.37/validators",
66 "http://207.261.33.37:8080/validators",
67 "https://ripple.com/validators",
68 "https://ripple.com:443/validators",
69 "file:///etc/opt/ripple/validators.txt",
70 "file:///C:/Lib/validators.txt"
71#if !_MSC_VER
72 ,
73 "file:///"
74#endif
75 });
76 BEAST_EXPECT(trustedSites->load(cfgSites));
77
78 // load should reject validator site uris with invalid schemes
79 std::vector<std::string> badSites({"ftp://ripple.com/validators"});
80 BEAST_EXPECT(!trustedSites->load(badSites));
81
82 badSites[0] = "wss://ripple.com/validators";
83 BEAST_EXPECT(!trustedSites->load(badSites));
84
85 badSites[0] = "ripple.com/validators";
86 BEAST_EXPECT(!trustedSites->load(badSites));
87
88 // Host names are not supported for file URLs
89 badSites[0] = "file://ripple.com/vl.txt";
90 BEAST_EXPECT(!trustedSites->load(badSites));
91
92 // Even local host names are not supported for file URLs
93 badSites[0] = "file://localhost/home/user/vl.txt";
94 BEAST_EXPECT(!trustedSites->load(badSites));
95
96 // Nor IP addresses
97 badSites[0] = "file://127.0.0.1/home/user/vl.txt";
98 BEAST_EXPECT(!trustedSites->load(badSites));
99
100 // File URL path can not be empty
101 badSites[0] = "file://";
102 BEAST_EXPECT(!trustedSites->load(badSites));
103
104#if _MSC_VER // Windows paths strip off the leading /, leaving the path empty
105 // File URL path can not be a directory
106 // (/ is the only path we can reasonably assume is a directory)
107 badSites[0] = "file:///";
108 BEAST_EXPECT(!trustedSites->load(badSites));
109#endif
110 }
111
125 void
127 detail::DirGuard const& good,
129 {
130 testcase << "Fetch list - "
131 << boost::algorithm::join(
132 paths |
133 boost::adaptors::transformed(
134 [](FetchListConfig const& cfg) {
135 return cfg.path +
136 (cfg.ssl ? " [https] v" : " [http] v") +
138 " " + cfg.msg;
139 }),
140 ", ");
141
142 using namespace jtx;
143 using namespace std::chrono_literals;
144
145 Env env(*this, [&]() {
146 auto p = test::jtx::envconfig();
147 p->legacy("database_path", good.subdir().string());
148 return p;
149 }());
150 auto& trustedKeys = env.app().validators();
151 env.timeKeeper().set(env.timeKeeper().now() + 30s);
152
153 test::StreamSink sink;
154 beast::Journal journal{sink};
155
156 std::vector<std::string> emptyCfgKeys;
157 struct publisher
158 {
159 publisher(FetchListConfig const& c) : cfg{c}
160 {
161 }
164 std::string uri;
165 FetchListConfig const& cfg;
166 bool isRetry;
167 };
169
170 auto constexpr listSize = 20;
171 std::vector<std::string> cfgPublishers;
172
173 for (auto const& cfg : paths)
174 {
175 servers.push_back(cfg);
176 auto& item = servers.back();
177 item.isRetry = cfg.path == "/bad-resource";
178 item.list.reserve(listSize);
179 while (item.list.size() < listSize)
180 item.list.push_back(TrustedPublisherServer::randomValidator());
181
182 NetClock::time_point const expires =
183 env.timeKeeper().now() + cfg.expiresFromNow;
184 NetClock::time_point const effective2 =
185 expires - cfg.effectiveOverlap;
186 NetClock::time_point const expires2 =
187 effective2 + cfg.expiresFromNow;
188 item.server = make_TrustedPublisherServer(
189 env.app().getIOContext(),
190 item.list,
191 expires,
192 {{effective2, expires2}},
193 cfg.ssl,
194 cfg.serverVersion);
195 std::string pubHex = strHex(item.server->publisherPublic());
196 cfgPublishers.push_back(pubHex);
197
198 if (item.cfg.failFetch)
199 {
200 // Create a cache file
201 auto const name = good.subdir() / ("cache." + pubHex);
202 std::ofstream o(name.string());
203 o << "{}";
204 }
205
207 uri << (cfg.ssl ? "https://" : "http://")
208 << item.server->local_endpoint() << cfg.path;
209 item.uri = uri.str();
210 }
211
212 BEAST_EXPECT(trustedKeys.load({}, emptyCfgKeys, cfgPublishers));
213
214 // Normally, tests will only need a fraction of this time,
215 // but sometimes DNS resolution takes an inordinate amount
216 // of time, so the test will just wait.
217 auto sites = std::make_unique<ValidatorSite>(env.app(), journal, 12s);
218
220 for (auto const& u : servers)
221 {
222 log << "Testing " << u.uri << std::endl;
223 uris.push_back(u.uri);
224 }
225 sites->load(uris);
226 sites->start();
227 sites->join();
228
229 auto const jv = sites->getJson();
230 for (auto const& u : servers)
231 {
232 for (auto const& val : u.list)
233 {
234 BEAST_EXPECT(
235 trustedKeys.listed(val.masterPublic) != u.cfg.failApply);
236 BEAST_EXPECT(
237 trustedKeys.listed(val.signingPublic) != u.cfg.failApply);
238 }
239
240 Json::Value myStatus;
241 for (auto const& vs : jv[jss::validator_sites])
242 if (vs[jss::uri].asString().find(u.uri) != std::string::npos)
243 myStatus = vs;
244 BEAST_EXPECTS(
245 myStatus[jss::last_refresh_message].asString().empty() !=
246 u.cfg.failFetch,
247 to_string(myStatus) + "\n" + sink.messages().str());
248
249 if (!u.cfg.msg.empty())
250 {
251 BEAST_EXPECTS(
252 sink.messages().str().find(u.cfg.msg) != std::string::npos,
253 sink.messages().str());
254 }
255
256 if (u.cfg.expectedRefreshMin)
257 {
258 BEAST_EXPECTS(
259 myStatus[jss::refresh_interval_min].asInt() ==
260 u.cfg.expectedRefreshMin,
261 to_string(myStatus));
262 }
263
264 if (u.cfg.failFetch)
265 {
266 using namespace std::chrono;
267 std::stringstream nextRefreshStr{
268 myStatus[jss::next_refresh_time].asString()};
269 system_clock::time_point nextRefresh;
270 date::from_stream(nextRefreshStr, "%Y-%b-%d %T", nextRefresh);
271 BEAST_EXPECT(!nextRefreshStr.fail());
272 auto now = system_clock::now();
273 BEAST_EXPECTS(
274 nextRefresh <= now + (u.isRetry ? seconds{30} : minutes{5}),
275 "Now: " + to_string(now) + ", NR: " + nextRefreshStr.str());
276 }
277 }
278 }
279
280 void
282 {
283 testcase << "File list - " << paths[0].first
284 << (paths.size() > 1 ? ", " + paths[1].first : "");
285
286 using namespace jtx;
287
288 Env env(*this);
289
290 test::StreamSink sink;
291 beast::Journal journal{sink};
292
293 struct publisher
294 {
295 std::string uri;
296 std::string expectMsg;
297 bool shouldFail;
298 };
300
301 for (auto const& cfg : paths)
302 {
303 servers.push_back({});
304 auto& item = servers.back();
305 item.shouldFail = !cfg.second.empty();
306 item.expectMsg = cfg.second;
307
309 uri << "file://" << cfg.first;
310 item.uri = uri.str();
311 }
312
313 auto sites = std::make_unique<ValidatorSite>(env.app(), journal);
314
316 for (auto const& u : servers)
317 uris.push_back(u.uri);
318 sites->load(uris);
319 sites->start();
320 sites->join();
321
322 for (auto const& u : servers)
323 {
324 auto const jv = sites->getJson();
325 Json::Value myStatus;
326 for (auto const& vs : jv[jss::validator_sites])
327 if (vs[jss::uri].asString().find(u.uri) != std::string::npos)
328 myStatus = vs;
329 BEAST_EXPECTS(
330 myStatus[jss::last_refresh_message].asString().empty() !=
331 u.shouldFail,
332 to_string(myStatus));
333 if (u.shouldFail)
334 {
335 BEAST_EXPECTS(
336 sink.messages().str().find(u.expectMsg) !=
337 std::string::npos,
338 sink.messages().str());
339 }
340 }
341 }
342
343 void
345 {
346 auto fullPath = [](detail::FileDirGuard const& guard) {
347 auto absPath = absolute(guard.file()).string();
348 if (absPath.front() != '/')
349 absPath.insert(absPath.begin(), '/');
350 return absPath;
351 };
352 {
353 // Create a file with a real validator list
355 *this, "test_val", "vl.txt", detail::realValidatorContents());
356 // Create a file with arbitrary content
358 *this, "test_val", "helloworld.txt", "Hello, world!");
359 // Create a file with malformed Json
361 *this,
362 "test_val",
363 "json.txt",
364 R"json({ "version": 2, "extra" : "value" })json");
365 auto const goodPath = fullPath(good);
366 auto const helloPath = fullPath(hello);
367 auto const jsonPath = fullPath(json);
368 auto const missingPath = jsonPath + ".bad";
369 testFileList({
370 {goodPath, ""},
371 {helloPath,
372 "Unable to parse JSON response from file://" + helloPath},
373 {jsonPath,
374 "Missing fields in JSON response from file://" + jsonPath},
375 {missingPath, "Problem retrieving from file://" + missingPath},
376 });
377 }
378 }
379
380public:
381 void
382 run() override
383 {
384 testConfigLoad();
385
386 detail::DirGuard good(*this, "test_fetch");
387 for (auto ssl : {true, false})
388 {
389 // fetch single site
390 testFetchList(good, {{"/validators", "", ssl}});
391 testFetchList(good, {{"/validators2", "", ssl}});
392 // fetch multiple sites
393 testFetchList(
394 good, {{"/validators", "", ssl}, {"/validators", "", ssl}});
395 testFetchList(
396 good, {{"/validators", "", ssl}, {"/validators2", "", ssl}});
397 testFetchList(
398 good, {{"/validators2", "", ssl}, {"/validators", "", ssl}});
399 testFetchList(
400 good, {{"/validators2", "", ssl}, {"/validators2", "", ssl}});
401 // fetch single site with single redirects
402 testFetchList(good, {{"/redirect_once/301", "", ssl}});
403 testFetchList(good, {{"/redirect_once/302", "", ssl}});
404 testFetchList(good, {{"/redirect_once/307", "", ssl}});
405 testFetchList(good, {{"/redirect_once/308", "", ssl}});
406 // one redirect, one not
407 testFetchList(
408 good,
409 {{"/validators", "", ssl}, {"/redirect_once/302", "", ssl}});
410 testFetchList(
411 good,
412 {{"/validators2", "", ssl}, {"/redirect_once/302", "", ssl}});
413 // UNLs with a "gap" between validUntil of one and validFrom of the
414 // next
415 testFetchList(
416 good,
417 {{"/validators2",
418 "",
419 ssl,
420 false,
421 false,
422 1,
424 std::chrono::seconds{-90}}});
425 // fetch single site with undending redirect (fails to load)
426 testFetchList(
427 good,
428 {{"/redirect_forever/301",
429 "Exceeded max redirects",
430 ssl,
431 true,
432 true}});
433 // two that redirect forever
434 testFetchList(
435 good,
436 {{"/redirect_forever/307",
437 "Exceeded max redirects",
438 ssl,
439 true,
440 true},
441 {"/redirect_forever/308",
442 "Exceeded max redirects",
443 ssl,
444 true,
445 true}});
446 // one undending redirect, one not
447 testFetchList(
448 good,
449 {{"/validators", "", ssl},
450 {"/redirect_forever/302",
451 "Exceeded max redirects",
452 ssl,
453 true,
454 true}});
455 // one undending redirect, one not
456 testFetchList(
457 good,
458 {{"/validators2", "", ssl},
459 {"/redirect_forever/302",
460 "Exceeded max redirects",
461 ssl,
462 true,
463 true}});
464 // invalid redir Location
465 testFetchList(
466 good,
467 {{"/redirect_to/ftp://invalid-url/302",
468 "Invalid redirect location",
469 ssl,
470 true,
471 true}});
472 testFetchList(
473 good,
474 {{"/redirect_to/file://invalid-url/302",
475 "Invalid redirect location",
476 ssl,
477 true,
478 true}});
479 // invalid json
480 testFetchList(
481 good,
482 {{"/validators/bad",
483 "Unable to parse JSON response",
484 ssl,
485 true,
486 true}});
487 testFetchList(
488 good,
489 {{"/validators2/bad",
490 "Unable to parse JSON response",
491 ssl,
492 true,
493 true}});
494 // error status returned
495 testFetchList(
496 good,
497 {{"/bad-resource", "returned bad status", ssl, true, true}});
498 // location field missing
499 testFetchList(
500 good,
501 {{"/redirect_nolo/308",
502 "returned a redirect with no Location",
503 ssl,
504 true,
505 true}});
506 // json fields missing
507 testFetchList(
508 good,
509 {{"/validators/missing",
510 "Missing fields in JSON response",
511 ssl,
512 true,
513 true}});
514 testFetchList(
515 good,
516 {{"/validators2/missing",
517 "Missing fields in JSON response",
518 ssl,
519 true,
520 true}});
521 // timeout
522 testFetchList(
523 good, {{"/sleep/13", "took too long", ssl, true, true}});
524 // bad manifest format using known versions
525 // * Retrieves a v1 formatted list claiming version 2
526 testFetchList(
527 good, {{"/validators", "Missing fields", ssl, true, true, 2}});
528 // * Retrieves a v2 formatted list claiming version 1
529 testFetchList(
530 good, {{"/validators2", "Missing fields", ssl, true, true, 0}});
531 // bad manifest version
532 // Because versions other than 1 are treated as v2, the v1
533 // list won't have the blobs_v2 fields, and thus will claim to have
534 // missing fields
535 testFetchList(
536 good, {{"/validators", "Missing fields", ssl, true, true, 4}});
537 testFetchList(
538 good,
539 {{"/validators2",
540 "1 unsupported version",
541 ssl,
542 false,
543 true,
544 4}});
545 using namespace std::chrono_literals;
546 // get expired validator list
547 testFetchList(
548 good,
549 {{"/validators",
550 "Applied 1 expired validator list(s)",
551 ssl,
552 false,
553 false,
554 1,
555 0s}});
556 testFetchList(
557 good,
558 {{"/validators2",
559 "Applied 1 expired validator list(s)",
560 ssl,
561 false,
562 false,
563 1,
564 0s,
565 -1s}});
566 // force an out-of-range validUntil value
567 testFetchList(
568 good,
569 {{"/validators",
570 "1 invalid validator list(s)",
571 ssl,
572 false,
573 true,
574 1,
576 // force an out-of-range validUntil value on the future list
577 // The first list is accepted. The second fails. The parser
578 // returns the "best" result, so this looks like a success.
579 testFetchList(
580 good,
581 {{"/validators2",
582 "",
583 ssl,
584 false,
585 false,
586 1,
588 299s}});
589 // force an out-of-range validFrom value
590 // The first list is accepted. The second fails. The parser
591 // returns the "best" result, so this looks like a success.
592 testFetchList(
593 good,
594 {{"/validators2",
595 "",
596 ssl,
597 false,
598 false,
599 1,
601 301s}});
602 // force an out-of-range validUntil value on _both_ lists
603 testFetchList(
604 good,
605 {{"/validators2",
606 "2 invalid validator list(s)",
607 ssl,
608 false,
609 true,
610 1,
613 // verify refresh intervals are properly clamped
614 testFetchList(
615 good,
616 {{"/validators/refresh/0",
617 "",
618 ssl,
619 false,
620 false,
621 1,
624 1}}); // minimum of 1 minute
625 testFetchList(
626 good,
627 {{"/validators2/refresh/0",
628 "",
629 ssl,
630 false,
631 false,
632 1,
635 1}}); // minimum of 1 minute
636 testFetchList(
637 good,
638 {{"/validators/refresh/10",
639 "",
640 ssl,
641 false,
642 false,
643 1,
646 10}}); // 10 minutes is fine
647 testFetchList(
648 good,
649 {{"/validators2/refresh/10",
650 "",
651 ssl,
652 false,
653 false,
654 1,
657 10}}); // 10 minutes is fine
658 testFetchList(
659 good,
660 {{"/validators/refresh/2000",
661 "",
662 ssl,
663 false,
664 false,
665 1,
668 60 * 24}}); // max of 24 hours
669 testFetchList(
670 good,
671 {{"/validators2/refresh/2000",
672 "",
673 ssl,
674 false,
675 false,
676 1,
679 60 * 24}}); // max of 24 hours
680 }
681 using namespace boost::filesystem;
682 for (auto const& file : directory_iterator(good.subdir()))
683 {
684 remove_all(file);
685 }
686
687 testFileURLs();
688 }
689};
690
691BEAST_DEFINE_TESTSUITE_PRIO(ValidatorSite, app, ripple, 2);
692
693} // namespace test
694} // namespace ripple
T back(T... args)
Represents a JSON value.
Definition json_value.h:130
std::string asString() const
Returns the unquoted string value.
static Int const maxInt
Definition json_value.h:143
A generic endpoint for log messages.
Definition Journal.h:41
A testsuite class.
Definition suite.h:52
log_os< char > log
Logging output stream.
Definition suite.h:149
testcase_t testcase
Memberspace for declaring test cases.
Definition suite.h:152
virtual ValidatorList & validators()=0
virtual boost::asio::io_context & getIOContext()=0
Create a directory and remove it when it's done.
path const & subdir() const
Write a file in a directory and remove when done.
time_point now() const override
Returns the current time.
std::stringstream const & messages() const
void testFetchList(detail::DirGuard const &good, std::vector< FetchListConfig > const &paths)
void run() override
Runs the suite.
void testFileList(std::vector< std::pair< std::string, std::string > > const &paths)
A transaction testing environment.
Definition Env.h:102
Application & app()
Definition Env.h:242
beast::Journal const journal
Definition Env.h:143
ManualTimeKeeper & timeKeeper()
Definition Env.h:254
Inject raw JSON.
Definition jtx_json.h:14
Set Paths, SendMax on a JTx.
Definition paths.h:16
T endl(T... args)
T is_same_v
constexpr char const * realValidatorContents()
auto constexpr default_expires
auto constexpr default_effective_overlap
std::unique_ptr< Config > envconfig()
creates and initializes a default configuration for jtx::Env
Definition envconfig.h:35
std::shared_ptr< TrustedPublisherServer > make_TrustedPublisherServer(boost::asio::io_context &ioc, std::vector< TrustedPublisherServer::Validator > const &validators, NetClock::time_point validUntil, std::vector< std::pair< NetClock::time_point, NetClock::time_point > > const &futures, bool useSSL=false, int version=1, bool immediateStart=true, int sequence=1)
Use hash_* containers for keys that do not need a cryptographically secure hashing algorithm.
Definition algorithm.h:6
std::string strHex(FwdIt begin, FwdIt end)
Definition strHex.h:11
std::string to_string(base_uint< Bits, Tag > const &a)
Definition base_uint.h:611
STL namespace.
T push_back(T... args)
T reserve(T... args)
T str(T... args)
T to_string(T... args)