2019-04-17 00:17:26 +02:00
|
|
|
/* This file is part of gitea2rss.
|
|
|
|
* Copyright © 2019 tastytea <tastytea@tastytea.de>
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU 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 General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <iostream>
|
2019-04-17 01:00:33 +02:00
|
|
|
#include <string>
|
|
|
|
#include <sstream>
|
|
|
|
#include <exception>
|
|
|
|
#include <cstdint>
|
2019-04-17 02:51:40 +02:00
|
|
|
#include <chrono>
|
|
|
|
#include <ctime>
|
2019-04-17 03:04:27 +02:00
|
|
|
#include <iomanip>
|
|
|
|
#include <sstream>
|
2019-04-17 05:52:10 +02:00
|
|
|
#include <cstdlib>
|
2019-04-17 01:00:33 +02:00
|
|
|
#include <curlpp/cURLpp.hpp>
|
|
|
|
#include <curlpp/Easy.hpp>
|
|
|
|
#include <curlpp/Options.hpp>
|
|
|
|
#include <curlpp/Exception.hpp>
|
|
|
|
#include <curlpp/Infos.hpp>
|
2019-04-17 01:23:23 +02:00
|
|
|
#include <jsoncpp/json/json.h>
|
2019-04-17 01:00:33 +02:00
|
|
|
#include "version.hpp"
|
2019-04-17 00:17:26 +02:00
|
|
|
|
|
|
|
using std::cout;
|
|
|
|
using std::cerr;
|
|
|
|
using std::endl;
|
2019-04-17 01:00:33 +02:00
|
|
|
using std::string;
|
|
|
|
using std::ostringstream;
|
2019-04-17 01:23:23 +02:00
|
|
|
using std::stringstream;
|
2019-04-17 09:13:01 +02:00
|
|
|
using std::uint8_t;
|
2019-04-17 01:00:33 +02:00
|
|
|
using std::uint16_t;
|
2019-04-17 02:51:40 +02:00
|
|
|
using std::chrono::system_clock;
|
2019-04-17 01:00:33 +02:00
|
|
|
|
|
|
|
namespace curlopts = curlpp::options;
|
|
|
|
|
2019-04-17 08:26:26 +02:00
|
|
|
bool cgi = false;
|
|
|
|
|
2019-04-17 02:51:40 +02:00
|
|
|
// Fetch HTTP document.
|
2019-04-17 01:00:33 +02:00
|
|
|
const string get_http(const string &url)
|
|
|
|
{
|
|
|
|
string answer;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
ostringstream oss;
|
|
|
|
curlpp::Easy request;
|
|
|
|
request.setOpt<curlopts::Url>(url);
|
|
|
|
request.setOpt<curlopts::UserAgent>(string("gitea2rss/")
|
|
|
|
+ global::version);
|
|
|
|
request.setOpt<curlopts::HttpHeader>({ "Connection: close" });
|
|
|
|
request.setOpt<curlopts::FollowLocation>(true);
|
|
|
|
request.setOpt<curlopts::WriteStream>(&oss);
|
|
|
|
request.perform();
|
|
|
|
uint16_t ret = curlpp::infos::ResponseCode::get(request);
|
|
|
|
if (ret == 200 || ret == 302 || ret == 307
|
|
|
|
|| ret == 301 || ret == 308)
|
|
|
|
{
|
|
|
|
answer = oss.str();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-04-17 08:26:26 +02:00
|
|
|
if (cgi)
|
|
|
|
{
|
|
|
|
cout << "Status: " << std::to_string(ret) << endl;
|
|
|
|
}
|
|
|
|
cerr << "HTTP Error: " << std::to_string(ret) << endl;
|
2019-04-17 01:00:33 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (const std::exception &e)
|
|
|
|
{
|
|
|
|
cerr << "Error: " << e.what() << endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
return answer;
|
|
|
|
}
|
2019-04-17 00:17:26 +02:00
|
|
|
|
2019-04-17 03:04:27 +02:00
|
|
|
// Convert time_point to RFC 822 compliant time string.
|
2019-04-17 02:51:40 +02:00
|
|
|
const string strtime(const system_clock::time_point &timepoint)
|
|
|
|
{
|
|
|
|
constexpr uint16_t bufsize = 1024;
|
|
|
|
std::time_t time = system_clock::to_time_t(timepoint);
|
2019-04-18 04:48:04 +02:00
|
|
|
std::tm *tm;
|
|
|
|
tm = std::gmtime(&time);
|
2019-04-17 02:51:40 +02:00
|
|
|
char buffer[bufsize];
|
2019-04-18 04:48:04 +02:00
|
|
|
std::strftime(buffer, bufsize, "%a, %d %b %Y %T %z", tm);
|
2019-04-17 02:51:40 +02:00
|
|
|
return static_cast<const string>(buffer);
|
|
|
|
}
|
|
|
|
|
2019-04-17 03:04:27 +02:00
|
|
|
// Convert ISO 8601 time string to RFC 822 time string.
|
|
|
|
const string strtime(const string &time)
|
|
|
|
{
|
|
|
|
std::tm tm = {};
|
2019-04-18 04:48:04 +02:00
|
|
|
tm.tm_isdst = -1; // Detect daylight saving time.
|
2019-04-17 03:04:27 +02:00
|
|
|
std::stringstream ss(time);
|
2019-04-18 04:48:04 +02:00
|
|
|
ss >> std::get_time(&tm, "%Y-%m-%dT%T"); // Assume time is UTC.
|
|
|
|
return strtime(std::chrono::system_clock::from_time_t(timegm(&tm)));
|
2019-04-17 03:04:27 +02:00
|
|
|
}
|
|
|
|
|
2019-04-17 09:13:01 +02:00
|
|
|
void write_line(const uint8_t spaces, const string &tag, const string &value)
|
|
|
|
{
|
|
|
|
string endtag;
|
|
|
|
// If there is a space in the tag, use only the part up until the space for
|
|
|
|
// the ending tag.
|
|
|
|
const size_t pos = tag.find(' ');
|
|
|
|
if (pos == std::string::npos)
|
|
|
|
{
|
|
|
|
endtag = tag;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
endtag = tag.substr(0, pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
cout << std::string(spaces, ' ');
|
|
|
|
cout << '<' << tag << '>' << value << "</" << endtag << ">\n";
|
|
|
|
}
|
|
|
|
|
2019-04-17 00:17:26 +02:00
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
2019-04-17 05:52:10 +02:00
|
|
|
const char *envquery = std::getenv("QUERY_STRING");
|
|
|
|
string url;
|
|
|
|
|
|
|
|
if (envquery != nullptr)
|
|
|
|
{
|
|
|
|
const string query = envquery;
|
|
|
|
const char *envbaseurl = std::getenv("GITEA2RSS_BASEURL");
|
|
|
|
if (envbaseurl == nullptr)
|
|
|
|
{
|
2019-04-17 08:26:26 +02:00
|
|
|
cout << "Status: 500 Internal Server Error\n\n";
|
2019-04-17 07:12:08 +02:00
|
|
|
cerr << "Error: GITEA2RSS_BASEURL not set\n";
|
2019-04-17 05:52:10 +02:00
|
|
|
return 1;
|
|
|
|
}
|
2019-04-17 08:26:26 +02:00
|
|
|
cgi = true;
|
|
|
|
|
2019-04-17 07:12:08 +02:00
|
|
|
const size_t pos = query.find("repo=");
|
2019-04-17 08:26:26 +02:00
|
|
|
if (pos == std::string::npos)
|
|
|
|
{
|
|
|
|
cout << "Status: 400 Bad Request\n\n";
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2019-04-17 06:34:48 +02:00
|
|
|
url = string(envbaseurl) + "/" + query.substr(query.find('=', pos) + 1);
|
2019-04-17 08:26:26 +02:00
|
|
|
cout << "Content-Type: application/rss+xml\n";
|
2019-04-17 05:52:10 +02:00
|
|
|
}
|
|
|
|
else if (argc < 2)
|
2019-04-17 00:17:26 +02:00
|
|
|
{
|
|
|
|
cerr << "usage: " << argv[0] << " URL of Gitea project\n";
|
|
|
|
return 1;
|
|
|
|
}
|
2019-04-17 05:52:10 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
url = argv[1];
|
|
|
|
}
|
2019-04-17 01:00:33 +02:00
|
|
|
|
|
|
|
curlpp::initialize();
|
|
|
|
|
|
|
|
size_t pos_repo = url.find('/', 8) + 1;
|
|
|
|
const string baseurl = url.substr(0, pos_repo - 1);
|
2019-04-17 02:51:40 +02:00
|
|
|
const string domain = baseurl.substr(baseurl.rfind('/') + 1);
|
2019-04-17 01:00:33 +02:00
|
|
|
const string repo = url.substr(pos_repo);
|
2019-04-17 02:51:40 +02:00
|
|
|
const string project = repo.substr(repo.find('/') + 1);
|
|
|
|
const string now = strtime(system_clock::now());
|
2019-04-17 01:23:23 +02:00
|
|
|
stringstream data(get_http(baseurl + "/api/v1/repos/"
|
|
|
|
+ repo + "/releases"));
|
2019-04-17 09:14:06 +02:00
|
|
|
if (cgi)
|
|
|
|
{
|
|
|
|
cout << endl;
|
|
|
|
}
|
2019-04-17 07:12:08 +02:00
|
|
|
if (data.str().empty())
|
|
|
|
{
|
|
|
|
cerr << "Error: Could not download releases.\n";
|
|
|
|
return 2;
|
|
|
|
}
|
2019-04-17 01:23:23 +02:00
|
|
|
Json::Value json;
|
|
|
|
data >> json;
|
2019-04-17 02:51:40 +02:00
|
|
|
|
|
|
|
cout <<
|
|
|
|
"<rss version=\"2.0\">\n"
|
|
|
|
" <channel>\n"
|
|
|
|
" <title>" << project << " releases</title>\n"
|
|
|
|
" <link>" << url << "</link>\n"
|
|
|
|
" <description>Releases of " << repo << "</description>\n"
|
|
|
|
" <generator>gitea2rss " << global::version << "</generator>\n"
|
|
|
|
" <lastBuildDate>" << now << "</lastBuildDate>\n";
|
|
|
|
|
2019-04-17 01:23:23 +02:00
|
|
|
for (const Json::Value &release : json)
|
|
|
|
{
|
2019-04-17 02:51:40 +02:00
|
|
|
const bool prerelease = release["prerelease"].asBool();
|
|
|
|
const string type = (prerelease ? "Pre-Release" : "Stable");
|
2019-04-17 09:13:01 +02:00
|
|
|
cout << " <item>\n";
|
|
|
|
write_line(6, "title", project + ": " + release["name"].asString());
|
|
|
|
write_line(6, "link", baseurl + "/" + repo + "/releases");
|
|
|
|
write_line(6, "guid isPermaLink=\"false\"",
|
|
|
|
domain + " release " + release["id"].asString());
|
|
|
|
write_line(6, "pubDate", strtime(release["published_at"].asString()));
|
|
|
|
write_line(6, "description",
|
2019-04-17 09:42:08 +02:00
|
|
|
"\n <![CDATA[<p><strong>" + type + "</strong></p>\n"
|
|
|
|
"<pre>" + release["body"].asString() + "</pre>\n"
|
|
|
|
" <p><a href=\"" + release["tarball_url"].asString()
|
|
|
|
+ "\">Download tarball</a></p>" + "]]>\n ");
|
2019-04-17 02:51:40 +02:00
|
|
|
cout << " </item>\n";
|
2019-04-17 01:23:23 +02:00
|
|
|
}
|
2019-04-17 01:00:33 +02:00
|
|
|
|
2019-04-17 02:51:40 +02:00
|
|
|
cout <<
|
|
|
|
" </channel>\n"
|
|
|
|
"</rss>\n";
|
|
|
|
|
2019-04-17 00:17:26 +02:00
|
|
|
return 0;
|
|
|
|
}
|