1
0
mirror of https://github.com/anope/anope.git synced 2026-06-12 17:04:47 +02:00
Files
2026-01-01 18:07:12 +00:00

194 lines
4.2 KiB
C++

// Anope IRC Services <https://www.anope.org/>
//
// Copyright (C) 2003-2026 Anope Contributors
//
// Anope is free software. You can use, modify, and/or distribute it under the
// terms of version 2 of the GNU General Public License. See docs/LICENSE.txt
// for the complete terms of this license and docs/AUTHORS.txt for a list of
// contributors.
//
// Based on the original code of Epona by Lara
// Based on the original code of Services by Andy Church
//
// SPDX-License-Identifier: GPL-2.0-only
#include <cstdlib>
#include <iostream>
#include <fstream>
#include <sstream>
#include <map>
static std::string get_git_hash(const std::string &git_dir)
{
std::fstream fd;
std::string filebuf;
fd.open((git_dir + "/HEAD").c_str(), std::ios::in);
if (!fd.is_open())
return "";
if (!getline(fd, filebuf) || filebuf.find("ref: ") != 0)
{
fd.close();
return "";
}
fd.close();
filebuf = filebuf.substr(5);
fd.open((git_dir + "/" + filebuf).c_str(), std::ios::in);
if (!fd.is_open())
return "";
if (!getline(fd, filebuf))
{
fd.close();
return "";
}
fd.close();
return filebuf.substr(0, 7);
}
static bool read_version_sh(const std::string &version_sh, std::map<std::string, std::string> &versions)
{
std::fstream fd(version_sh.c_str(), std::ios::in);
if (!fd.is_open())
{
std::cerr << "Error: Unable to open src/version.sh for reading: " << version_sh << std::endl;
return false;
}
std::string filebuf;
while (getline(fd, filebuf))
{
if (!filebuf.find("VERSION_"))
{
size_t eq = filebuf.find('=');
std::string type = filebuf.substr(0, eq);
std::string value = filebuf.substr(eq + 1);
versions[type] = value;
}
}
fd.close();
return true;
}
static bool write_build_h(const std::string &buildh, const std::string &git_version)
{
std::fstream fd(buildh.c_str(), std::ios::in);
std::string build = "#define BUILD 1";
if (fd.is_open())
{
for (std::string filebuf; getline(fd, filebuf);)
{
if (!filebuf.find("#define BUILD"))
{
size_t tab = filebuf.find(' ');
int ibuild = atoi(filebuf.substr(tab + 1).c_str()) + 1;
std::stringstream ss;
ss << "#define BUILD " << ibuild;
build = ss.str();
}
}
fd.close();
}
fd.clear();
fd.open(buildh.c_str(), std::ios::out);
if (!fd.is_open())
{
std::cerr << "Error: Unable to open build.h for writing: " << buildh << std::endl;
return false;
}
fd << "/* This file is automatically generated by version.cpp - do not edit it! */" << std::endl
<< "#pragma once" << std::endl
<< build << std::endl;
if (!git_version.empty())
fd << "#define VERSION_GIT \"" << git_version << "\"" << std::endl;
fd.close();
return true;
}
static void read_version_h(const std::string &versionh, std::map<std::string, std::string> &versions)
{
std::fstream fd(versionh.c_str(), std::ios::in);
if (!fd.is_open())
return;
for (std::string filebuf; getline(fd, filebuf);)
{
if (!filebuf.find("#define VERSION_"))
{
size_t space = filebuf.substr(8).find(' ');
std::string name = filebuf.substr(8).substr(0, space),
version = filebuf.substr(8).substr(space + 1);
versions[name] = version;
}
}
fd.close();
}
static bool write_version_h(const std::string &versionh, const std::map<std::string, std::string> &versions)
{
std::fstream fd(versionh.c_str(), std::ios::out);
if (!fd.is_open())
return false;
fd << "#pragma once" << std::endl;
for (const auto &[key, value] : versions)
{
fd << "#define " << key << " " << value << std::endl;
}
fd.close();
return true;
}
int main(int argc, char *argv[])
{
if (argc < 4)
{
std::cerr << "Syntax: " << argv[0] << " <base> <version.h> <build.h>" << std::endl;
return 1;
}
std::string version_sh = std::string(argv[1]) + "/src/version.sh";
std::string git_dir = std::string(argv[1]) + "/.git";
std::string versionh = argv[2];
std::string buildh = argv[3];
std::map<std::string, std::string> versions, old_versions;
if (!read_version_sh(version_sh, versions))
return EXIT_FAILURE;
std::string git_version = get_git_hash(git_dir);
if (!write_build_h(buildh, git_version))
return EXIT_FAILURE;
read_version_h(versionh, old_versions);
if (versions == old_versions)
return EXIT_SUCCESS;
if (!write_version_h(versionh, versions))
return EXIT_FAILURE;
return EXIT_SUCCESS;
}