/* Build bumper * * (C) 2003-2022 Anope Team * Contact us at team@anope.org * * Please read COPYING and README for further details. * * Based on the original code of Epona by Lara. * Based on the original code of Services by Andy Church. */ #include #include #include #include #include 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 "g" + filebuf.substr(0, 7); } static bool read_version_sh(const std::string &version_sh, std::map &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; fd << 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 &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 &versions) { std::fstream fd(versionh.c_str(), std::ios::out); if (!fd.is_open()) return false; for (std::map::const_iterator it = versions.begin(); it != versions.end(); ++it) { fd << "#define " << it->first << " " << it->second << std::endl; } fd.close(); return true; } int main(int argc, char *argv[]) { if (argc < 4) { std::cerr << "Syntax: " << argv[0] << " " << 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 versions, old_versions; if (!read_version_sh(version_sh, versions)) return -1; std::string git_version = get_git_hash(git_dir); if (!write_build_h(buildh, git_version)) return -1; read_version_h(versionh, old_versions); if (versions == old_versions) return 0; if (!write_version_h(versionh, versions)) return -1; return 0; }