3#include <test/jtx/envconfig.h>
5#include <xrpl/basics/base64.h>
6#include <xrpl/basics/random.h>
7#include <xrpl/basics/strHex.h>
8#include <xrpl/protocol/PublicKey.h>
9#include <xrpl/protocol/SecretKey.h>
10#include <xrpl/protocol/Sign.h>
12#include <boost/algorithm/string/predicate.hpp>
13#include <boost/asio.hpp>
14#include <boost/asio/ip/tcp.hpp>
15#include <boost/asio/ssl/stream.hpp>
16#include <boost/beast/core/flat_buffer.hpp>
17#include <boost/beast/http.hpp>
18#include <boost/beast/ssl.hpp>
19#include <boost/beast/version.hpp>
20#include <boost/lexical_cast.hpp>
34 using req_type = boost::beast::http::request<boost::beast::http::string_body>;
35 using resp_type = boost::beast::http::response<boost::beast::http::string_body>;
50 boost::asio::ssl::context
sslCtx_{boost::asio::ssl::context::tlsv12};
61 [](
std::size_t, boost::asio::ssl::context_base::password_purpose) {
return "test"; });
64 boost::asio::ssl::context::default_workarounds | boost::asio::ssl::context::no_sslv2 |
65 boost::asio::ssl::context::single_dh_use);
67 sslCtx_.use_certificate_chain(boost::asio::buffer(
cert().data(),
cert().size()));
70 boost::asio::buffer(
key().data(),
key().size()),
71 boost::asio::ssl::context::file_format::pem);
73 sslCtx_.use_tmp_dh(boost::asio::buffer(
dh().data(),
dh().size()));
105 st[sfSequence] = seq;
106 st[sfPublicKey] = pk;
107 st[sfSigningPubKey] = spk;
128 .masterPublic = masterPublic,
129 .signingPublic = signingKeys.first,
141 boost::asio::io_context& ioc,
147 bool immediateStart =
true,
159 auto const manifest =
163 blobInfo.
reserve(futures.size() + 1);
170 for (
auto const& val : validators)
172 data +=
"{\"validation_public_key\":\"" +
strHex(val.masterPublic) +
173 "\",\"manifest\":\"" + val.manifest +
"\"},";
182 getList_ = [blob = blob, sig, manifest, version](
int interval) {
185 l <<
"{\"blob\":\"" << blob <<
"\"" <<
",\"signature\":\"" << sig <<
"\""
186 <<
",\"manifest\":\"" << manifest <<
"\""
187 <<
",\"refresh_interval\": " << interval <<
",\"version\":" << version <<
'}';
190 for (
auto const& future : futures)
193 ",\"effective\":" +
std::to_string(future.first.time_since_epoch().count()) +
194 ",\"expiration\":" +
std::to_string(future.second.time_since_epoch().count()) +
198 for (
auto const& val : validators)
200 data +=
"{\"validation_public_key\":\"" +
strHex(val.masterPublic) +
201 "\",\"manifest\":\"" + val.manifest +
"\"},";
209 getList2_ = [blobInfo, manifest, version](
int interval) {
214 for (
auto const& info : blobInfo)
216 l <<
"{\"blob\":\"" << info.blob <<
"\"" <<
",\"signature\":\"" << info.signature
222 l <<
"{\"blobs_v2\": [ " << blobs <<
"],\"manifest\":\"" << manifest <<
"\""
223 <<
",\"refresh_interval\": " << interval <<
",\"version\":" << (version + 1) <<
'}';
239 acceptor_.set_option(boost::asio::ip::tcp::acceptor::reuse_address(
true), ec);
241 acceptor_.listen(boost::asio::socket_base::max_listen_connections);
244 if (
auto p = wp.lock())
343-----BEGIN CERTIFICATE-----
344MIIDczCCAlugAwIBAgIBATANBgkqhkiG9w0BAQsFADBjMQswCQYDVQQGEwJVUzEL
345MAkGA1UECAwCQ0ExFDASBgNVBAcMC0xvcyBBbmdlbGVzMRswGQYDVQQKDBJyaXBw
346bGVkLXVuaXQtdGVzdHMxFDASBgNVBAMMC2V4YW1wbGUuY29tMB4XDTIyMDIwNTIz
347NDk0M1oXDTQ5MDYyMzIzNDk0M1owazELMAkGA1UEBhMCVVMxEzARBgNVBAgMCkNh
348bGlmb3JuaWExFjAUBgNVBAcMDVNhbiBGcmFuY2lzY28xGzAZBgNVBAoMEnJpcHBs
349ZWQtdW5pdC10ZXN0czESMBAGA1UEAwwJMTI3LjAuMC4xMIIBIjANBgkqhkiG9w0B
350AQEFAAOCAQ8AMIIBCgKCAQEAueZ1hgRxwPgfeVx2AdngUYx7zYcaxcGYXyqi7izJ
351qTuBUcVcTRC/9Ip67RAEhfcgGudRS/a4Sv1ljwiRknSCcD/ZjzOFDLgbqYGSZNEs
352+T/qkwmc/L+Pbzf85HM7RjeGOd6NDQy9+oOBbUtqpTxcSGa4ln+YBFUSeoS1Aa9f
353n9vrxnWX9LgTu5dSWzH5TqFIti+Zs/v0PFjEivBIAOHPslmnzg/wCr99I6z9CAR3
354zVDe7+sxR//ivpeVE7FWjgkGixnUpZAqn69zNkJjMLNXETgOYskZdMIgbVOMr+0q
355S1Uj77mhwxKfpnB6TqUVvWLBvmBDzPjf0m0NcCf9UAjqPwIDAQABoyowKDAmBgNV
356HREEHzAdgglsb2NhbGhvc3SHEAAAAAAAAAAAAAAAAAAAAAEwDQYJKoZIhvcNAQEL
357BQADggEBAJkUFNS0CeEAKvo0ttzooXnCDH3esj2fwmLJQYLUGsAF8DFrFHTqZEcx
358hFRdr0ftEb/VKpV9dVF6xtSoMU56kHOnhbHEWADyqdKUkCDjrGBet5QdWmEwNV2L
359nYrwGQBAybMt/+1XMUV8HeLFJNHnyxfQYcW0fUsrmNGk8W0kzWuuq88qbhfXZAIx
360KiXrzYpLlM0RlpWXRfYQ6mTdSrRrLnEo5MklizVgNB8HYX78lxa06zP08oReQcfT
361GSGO8NEEq8BTVmp69zD1JyfvQcXzsi7WtkAX+/EOFZ7LesnZ6VsyjZ74wECCaQuD
362X1yu/XxHqchM+DOzzVw6wRKaM7Zsk80=
363-----END CERTIFICATE-----
372-----BEGIN RSA PRIVATE KEY-----
373MIIEpAIBAAKCAQEAueZ1hgRxwPgfeVx2AdngUYx7zYcaxcGYXyqi7izJqTuBUcVc
374TRC/9Ip67RAEhfcgGudRS/a4Sv1ljwiRknSCcD/ZjzOFDLgbqYGSZNEs+T/qkwmc
375/L+Pbzf85HM7RjeGOd6NDQy9+oOBbUtqpTxcSGa4ln+YBFUSeoS1Aa9fn9vrxnWX
3769LgTu5dSWzH5TqFIti+Zs/v0PFjEivBIAOHPslmnzg/wCr99I6z9CAR3zVDe7+sx
377R//ivpeVE7FWjgkGixnUpZAqn69zNkJjMLNXETgOYskZdMIgbVOMr+0qS1Uj77mh
378wxKfpnB6TqUVvWLBvmBDzPjf0m0NcCf9UAjqPwIDAQABAoIBAEC9MDpOu+quvg8+
379kt4MKSFdIhQuM7WguNaTe5AkSspDrcJzT7SK275mp259QIYCzMxxuA8TSZTb8A1C
380t6dgKbi7k6FaGMCYMRHzzK6NZfMbPi6cj245q9LYlZpdQswuM/FdPpPH1zUxrNYK
381CIaooZ6ZHzlSD/eaRMgkBQEkONHrZZtEinLIvKedwssPCaXkIISmt7MFQTDOlxkf
382K0Mt1mnRREPYbYSfPEEfIyy/KDIiB5AzgGt+uPOn8Oeb1pSqy69jpYcfhSj+bo4S
383UV6qTuTfBd4qkkNI6d/Z7DcDJFFlfloG/vVgGk/beWNnL2e39vzxiebB3w+MQn4F
384Wyx5mCECgYEA22z1/ihqt9LIAWtP42oSS3S/RxlFzpp5d7QfNqFnEoVgeRhQzleP
385pRJIzVXpMYBxexZYqZA/q8xBSggz+2gmRoYnW20VIzl14DsSH378ye3FRwJB0tLy
386dWU8DC7ZB5XQCTvI9UY3voJNToknODw7RCNO1h3V3T1y6JRLdcLskk8CgYEA2OLy
387aE5bvsUaLBSv7W9NFhSuZ0p9Y0pFmRgHI7g8i/AgRZ0BgiE8u8OZSHmPJPMaNs/h
388YIEIrlsgDci1PzwrUYseRp/aiVE1kyev09/ihqRXTPpLQu6h/d63KRe/06W3t5X3
389Dmfj49hH5zGPBI/0y1ECV/n0fwnRhxSv7fNr3RECgYBEuFpOUAAkNApZj29ErNqv
3908Q9ayAp5yx1RpQLFjEUIoub05e2gwgGF1DUiwc43p59iyjvYVwnp1x13fxwwl4yt
391N6Sp2H7vOja1lCp33MB0yVeohodw7InsxFjLA/0KiBvQWH32exhIPOzTNNcooIx7
392KYeuPUfWc0FCn/cGGZcXtwKBgQC1hp1k99CKBuY05suoanOWe5DNGud/ZvaBgD7Z
393gqYKadxY52QPyknOzZNJuZQ5VM8n+S2lW9osNFDLuKUaW/3Vrh6U9c4vCC1TEPB0
3944PnzvzDiWMsNJjWnCfU7C4meVyFBIt84y3NNjAQCWNRe+S3lzdOsVqRwf4NDD+l/
395uzEYQQKBgQCJczIlwobm1Y6O41hbGZhZL/CGMNS6Z0INi2yasV0WDqYlh7XayHMD
396cK55dMILcbHqeIBq/wR6sIhw6IJcaDBfFfrJiKKDilfij2lHxR2FQrEngtTCCRV+
397ZzARzaWhQPvbDqEtLJDWuXZNXfL8/PTIs5NmuKuQ8F4+gQJpkQgwaw==
398-----END RSA PRIVATE KEY-----
407-----BEGIN CERTIFICATE-----
408MIIDpzCCAo+gAwIBAgIUWc45WqaaNuaSLoFYTMC/Mjfqw/gwDQYJKoZIhvcNAQEL
409BQAwYzELMAkGA1UEBhMCVVMxCzAJBgNVBAgMAkNBMRQwEgYDVQQHDAtMb3MgQW5n
410ZWxlczEbMBkGA1UECgwScmlwcGxlZC11bml0LXRlc3RzMRQwEgYDVQQDDAtleGFt
411cGxlLmNvbTAeFw0yMjAyMDUyMzQ5MDFaFw00OTA2MjMyMzQ5MDFaMGMxCzAJBgNV
412BAYTAlVTMQswCQYDVQQIDAJDQTEUMBIGA1UEBwwLTG9zIEFuZ2VsZXMxGzAZBgNV
413BAoMEnJpcHBsZWQtdW5pdC10ZXN0czEUMBIGA1UEAwwLZXhhbXBsZS5jb20wggEi
414MA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQC0f2JBW2XNW2wT5/ajX2qxmUY+
415aNJGfpV6gZ5CmwdQpbHrPPvJoskxwsCyr3GifzT/GtCpmb1fiu59uUAPxQEYCxiq
416V+HchX4g4Vl27xKJ0P+usxuEED9v7TCteKum9u9eMZ8UDF0fspXcnWGs9fXlyoTj
417uTRP1SBQllk44DPc/KzlrtH+QNXmr9XQnP8XvwWCgJXMx87voxEGiFFOVhkSSAOv
418v+OUGgEuq0NPgwv2LHBlYHSdkoU9F5Z/TmkCAFMShbyoUjldIz2gcWXjN2tespGo
419D6qYvasvPIpmcholBBkc0z8QDt+RNq+Wzrults7epJXy/u+txGK9cHCNlLCpAgMB
420AAGjUzBRMB0GA1UdDgQWBBS1oydh+YyqDNOFKYOvOtVMWKqV4zAfBgNVHSMEGDAW
421gBS1oydh+YyqDNOFKYOvOtVMWKqV4zAPBgNVHRMBAf8EBTADAQH/MA0GCSqGSIb3
422DQEBCwUAA4IBAQCDPyGKQwQ8Lz0yEgvIl/Uo9BtwAzlvjrLM/39qhStLQqDGSs2Q
423xFIbtjzjuLf5vR3q6OJ62CCvzqXgHkJ+hzVN/tAvyliGTdjJrK+xv1M5a+XipO2f
424c9lb4gRbFL/DyoeoWgb1Rkv3gFf0FlCYH+ZUcYb9ZYCRlGtFgOcxJI2g+T7jSLFp
4258+hSzQ6W5Sp9L6b5iJyCww1vjBvBqzNyZMNeB4gXGtd6z9vMDSvKboTdGD7wcFB+
426mRMyNekaRw+Npy4Hjou5sx272cXHHmPCSF5TjwdaibSaGjx1k0Q50mOf7S9KG5b5
4277X1e3FekJlaD02EBEhtkXURIxogOQALdFncj
428-----END CERTIFICATE-----
437-----BEGIN DH PARAMETERS-----
438MIIBCAKCAQEAp2I2fWEUZ3sCNfitSRC/MdAhJE/bS+NO0O2tWdIdlvmIFE6B5qhC
439sGW9ojrQT8DTxBvGAcbjr/jagmlE3BV4oSnxyhP37G2mDvMOJ29J3NvFD/ZFAW0d
440BvZJ1RNvMu29NmVCyt6/jgzcqrqnami9uD93aK+zaVrlPsPEYM8xB19HXwqsEYCL
441ux2B7sqXm9Ts74HPg/EV+pcVon9phxNWxxgHlOvFc2QjZ3hXH++kzmJ4vs7N/XDB
442xbEQ+TUZ5jbJGSeBqNFKFeuOUQGJ46Io0jBSYd4rSmKUXkvElQwR+n7KF3jy1uAt
443/8hzd8tHn9TyW7Q2/CPkOA6dCXzltpOSowIBAg==
444-----END DH PARAMETERS-----
455 boost::asio::executor_work_guard<boost::asio::executor>
work;
476 static int nextId = 0;
480 if (
auto p = wp.lock())
490 using namespace boost::beast;
491 using namespace boost::asio;
495 std::optional<ssl_stream<ip::tcp::socket&>> sslStream;
502 sslStream->handshake(ssl::stream_base::server, ec);
516 *sslStream, sb, req, ec);
521 http::read(sock, sb, req, ec);
527 std::string_view
const path = req.target();
528 res.insert(
"Server",
"TrustedPublisherServer");
529 res.version(req.version());
530 res.keep_alive(req.keep_alive());
533 if (boost::starts_with(path,
"/validators2"))
535 res.result(http::status::ok);
536 res.insert(
"Content-Type",
"application/json");
537 if (path ==
"/validators2/bad")
539 res.body() =
"{ 'bad': \"2']";
541 else if (path ==
"/validators2/missing")
543 res.body() =
"{\"version\": 2}";
548 static constexpr char const* kRefreshPrefix =
"/validators2/refresh/";
549 if (boost::starts_with(path, kRefreshPrefix))
551 refresh = boost::lexical_cast<unsigned int>(
557 else if (boost::starts_with(path,
"/validators"))
559 res.result(http::status::ok);
560 res.insert(
"Content-Type",
"application/json");
561 if (path ==
"/validators/bad")
563 res.body() =
"{ 'bad': \"1']";
565 else if (path ==
"/validators/missing")
567 res.body() =
"{\"version\": 1}";
572 static constexpr char const* kRefreshPrefix =
"/validators/refresh/";
573 if (boost::starts_with(path, kRefreshPrefix))
575 refresh = boost::lexical_cast<unsigned int>(
581 else if (boost::starts_with(path,
"/textfile"))
584 res.result(http::status::ok);
585 res.insert(
"Content-Type",
"text/example");
587 std::uint64_t
const cl = boost::starts_with(path,
"/textfile/huge")
590 res.content_length(cl);
591 if (req.method() == http::verb::get)
593 std::stringstream body;
594 for (
auto i = 0; i < 1024; ++i)
597 res.body() = body.
str();
601 else if (boost::starts_with(path,
"/sleep/"))
603 auto const sleepSec = boost::lexical_cast<unsigned int>(path.
substr(7));
606 else if (boost::starts_with(path,
"/redirect"))
608 if (boost::ends_with(path,
"/301"))
610 res.result(http::status::moved_permanently);
612 else if (boost::ends_with(path,
"/302"))
614 res.result(http::status::found);
616 else if (boost::ends_with(path,
"/307"))
618 res.result(http::status::temporary_redirect);
620 else if (boost::ends_with(path,
"/308"))
622 res.result(http::status::permanent_redirect);
625 std::stringstream location;
626 if (boost::starts_with(path,
"/redirect_to/"))
628 location << path.
substr(13);
630 else if (!boost::starts_with(path,
"/redirect_nolo"))
632 location << (ssl ?
"https://" :
"http://") <<
localEndpoint()
633 << (boost::starts_with(path,
"/redirect_forever/")
637 if (!location.
str().empty())
638 res.insert(
"Location", location.
str());
643 res.result(boost::beast::http::status::not_found);
644 res.insert(
"Content-Type",
"text/html");
645 res.body() =
"The file '" + std::string(path) +
651 res.prepare_payload();
653 catch (std::exception
const& e)
656 res.result(boost::beast::http::status::internal_server_error);
657 res.version(req.version());
658 res.insert(
"Server",
"TrustedPublisherServer");
659 res.insert(
"Content-Type",
"text/html");
660 res.body() = std::string{
"An internal error occurred"} + e.
what();
661 res.prepare_payload();
666 write(*sslStream, res, ec);
671 write(sock, res, ec);
674 if (ec || req.need_eof())
680 sslStream->shutdown(ec);
685inline std::shared_ptr<TrustedPublisherServer>
687 boost::asio::io_context& ioc,
693 bool immediateStart =
true,
697 ioc, validators, validUntil, futures, useSSL, version, sequence);
std::chrono::time_point< NetClock > time_point
void add(Serializer &s) const override
std::size_t size() const noexcept
void const * data() const noexcept
boost::asio::ip::address address_type
static std::string const & dh()
static std::string makeManifestString(PublicKey const &pk, SecretKey const &sk, PublicKey const &spk, SecretKey const &ssk, int seq)
PublicKey const & publisherPublic() const
boost::beast::http::request< boost::beast::http::string_body > req_type
boost::asio::ssl::context sslCtx_
static std::string const & caCert()
PublicKey publisherPublic_
static std::string const & key()
boost::asio::ip::tcp::acceptor acceptor_
void onAccept(error_code ec)
~TrustedPublisherServer()
static std::string const & cert()
static Validator randomValidator()
std::function< std::string(int)> getList_
endpoint_type localEndpoint() const
SecretKey publisherSecret_
TrustedPublisherServer(boost::asio::io_context &ioc, std::vector< 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)
boost::asio::ip::tcp::endpoint endpoint_type
boost::system::error_code error_code
void doPeer(int id, socket_type &&s, bool ssl)
boost::asio::ip::tcp::socket socket_type
boost::beast::http::response< boost::beast::http::string_body > resp_type
std::function< std::string(int)> getList2_
void loadServerCertificate()
T emplace_back(T... args)
void write(nudb::detail::ostream &os, std::size_t t)
void sign(json::Value &jv, Account const &account, json::Value &sigObject)
Sign automatically into a specific Json field of the jv object.
std::shared_ptr< TrustedPublisherServer > makeTrustedPublisherServer(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)
char const * getEnvLocalhostAddr()
Use hash_* containers for keys that do not need a cryptographically secure hashing algorithm.
std::pair< PublicKey, SecretKey > randomKeyPair(KeyType type)
Create a key pair using secure random numbers.
PublicKey derivePublicKey(KeyType type, SecretKey const &sk)
Derive the public key from a secret key.
std::string strHex(FwdIt begin, FwdIt end)
std::enable_if_t< std::is_integral_v< Integral >, Integral > randInt()
std::optional< KeyType > publicKeyType(Slice const &slice)
Returns the type of public key.
SecretKey randomSecretKey()
Create a secret key using secure random numbers.
std::string base64Encode(std::uint8_t const *data, std::size_t len)
std::enable_if_t< std::is_same_v< T, char >||std::is_same_v< T, unsigned char >, Slice > makeSlice(std::array< T, N > const &a)
T shared_from_this(T... args)
BlobInfo(std::string b, std::string s)
Lambda(int id, TrustedPublisherServer &self, socket_type &&sock, bool ssl)
TrustedPublisherServer & self
boost::asio::executor_work_guard< boost::asio::executor > work
T time_since_epoch(T... args)