This repository has been archived on 2021-03-22. You can view files and clone it, but cannot push or open issues or pull requests.
backend/src/generators/rss.cpp

222 lines
6.1 KiB
C++

/* This file is part of FediBlock-backend.
* Copyright © 2020 tastytea <tastytea@tastytea.de>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, version 3.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "rss.hpp"
#include "files.hpp"
#include "git.hpp"
#include "time.hpp"
#include <algorithm>
#include <chrono>
#include <cstdint>
#include <cstdlib>
#include <exception>
#include <iostream>
#include <ostream>
#include <string>
#include <string_view>
#include <vector>
namespace FediBlock::rss
{
using std::find;
using std::getenv;
using std::none_of;
using std::ostream;
using std::string;
using std::string_view;
using std::uint8_t;
using std::chrono::system_clock;
void write_line(ostream &out, const uint8_t spaces, const string_view rsstag,
const string_view value)
{
// clang-format off
const string endtag{[&rsstag]
{
// If there is a space in the rsstag, use only the part up until the
// space for the ending RSS tag.
const size_t pos = rsstag.find(' ');
if (pos == string_view::npos)
{
return rsstag;
}
return rsstag.substr(0, pos);
}()};
// clang-format off
out << string(spaces, ' ');
out << '<' << rsstag << '>' << value << "</" << endtag << ">\n";
}
void write_rss(ostream &out, const vector<entry_type> &entries,
const vector<string> &tags)
{
string selfurl;
char *env{getenv("HTTPS")};
if (env == nullptr || string(env) != "on")
{
selfurl = "http://";
}
else
{
selfurl = "https://";
}
env = getenv("SERVER_NAME");
if (env != nullptr)
{
selfurl += env;
}
const string baseurl{selfurl + "/blocklist/"};
env = getenv("REQUEST_URI");
if (env != nullptr)
{
selfurl += env;
}
constexpr string_view blocklist_url{
"https://fediblock-test.tastytea.de/blocklist/"};
constexpr string_view rss_time_format{"%a, %d %b %Y %T %z"};
out << R"(<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom">
<channel>
)";
out << R"( <atom:link href=")" + selfurl
<< R"(" rel="self" type="application/rss+xml"/>)" << '\n';
write_line(out, 4, "title", "FediBlock: newest entries");
write_line(out, 4, "link", blocklist_url);
string description;
if (tags.empty())
{
description = "All tags.";
}
else
{
description = "Tags: ";
}
for (const auto &tag : tags)
{
if (tag == *(tags.begin()))
{
description += ", ";
}
description += tag;
}
write_line(out, 4, "description", description);
write_line(out, 4, "lastBuildDate",
time::to_string(system_clock::now(), rss_time_format));
for (const auto &entry : entries)
{
// If tags are specified and none of them match, go to the next entry.
if (!tags.empty())
{
// clang-format off
if (none_of(entry.tags.begin(), entry.tags.end(),
[&tags](const auto &tag)
{
return find(tags.begin(), tags.end(), tag)
!= tags.end();
}))
{
continue;
}
// clang-format on
}
out << " <item>\n";
write_line(out, 6, "title", entry.instance);
write_line(out, 6, "guid isPermaLink=\"false\"",
"FediBlock: " + entry.report_time + " " + entry.instance);
write_line(out, 6, "pubDate",
time::to_string(entry.report_time, rss_time_format));
string item_description{"<p>" + entry.description + "</p>"};
item_description += "<p><strong>Tags:</strong> ";
for (const auto &tag : entry.tags)
{
if (tag != *(entry.tags.begin()))
{
item_description += ", ";
}
item_description += tag;
}
item_description += "</p>";
item_description += "<p><strong>Receipts:</strong><br>";
for (const auto &receipt : entry.receipts)
{
if (receipt != *(entry.receipts.begin()))
{
item_description += "<br>";
}
((((item_description += "<a href=\"") += receipt) += "\">") +=
receipt) += "</a>";
}
item_description += "</p>";
if (!entry.screenshot_filepath.empty())
{
item_description += "<p><strong>Screenshot:</strong><br>";
((item_description += "<a href=\"") += baseurl) +=
entry.screenshot_filepath;
item_description += "\"><img src=\"";
(item_description += baseurl) += entry.screenshot_filepath;
(item_description += R"(" height="100"></a></p>)") += '\n';
}
write_line(out, 6, "description",
"<![CDATA[" + item_description + "]]>");
out << " </item>\n";
}
out << " </channel>\n</rss>\n";
}
} // namespace FediBlock::rss
using std::cerr;
using std::cout;
using std::exception;
using std::string_view;
using std::vector;
using namespace FediBlock;
using namespace FediBlock::rss;
int main(int argc, char *argv[])
{
const vector<string_view> args(argv, argv + argc);
git_libgit2_init();
try
{
git::clone();
const auto entries{files::read_json_files()};
cout << "Content-Type: application/rss+xml\r\n\r\n";
write_rss(cout, entries, {});
}
catch (const exception &e)
{
cerr << "Error: " << e.what() << '\n';
}
files::remove_tmpdir();
git_libgit2_shutdown();
return 0;
}