Compare commits
4 Commits
v2025.0626
...
v2025.0629
Author | SHA1 | Date | |
---|---|---|---|
50fb5f9da6 | |||
8e2611e362 | |||
a1b12fe177 | |||
902e68069a |
@ -36,13 +36,16 @@ target_include_directories(${PROJECT_NAME} PRIVATE
|
||||
src/common)
|
||||
|
||||
# Find packages
|
||||
find_package(OpenSSL REQUIRED)
|
||||
find_package(Drogon CONFIG REQUIRED)
|
||||
find_package(nlohmann_json REQUIRED)
|
||||
|
||||
# Add module path for FindCPRStatic
|
||||
list(APPEND CMAKE_MODULE_PATH "/usr/local/share/cmake/Modules")
|
||||
|
||||
# Find packages
|
||||
find_package(nlohmann_json REQUIRED)
|
||||
find_package(CPRStatic REQUIRED)
|
||||
|
||||
# Link libraries
|
||||
target_link_libraries(${PROJECT_NAME} PRIVATE
|
||||
nlohmann_json::nlohmann_json Drogon::Drogon
|
||||
/usr/local/lib/libpgcommon.a /usr/local/lib/libpgport.a
|
||||
lzma dl)
|
||||
|
||||
nlohmann_json::nlohmann_json
|
||||
cpr::cpr_static)
|
1
getpkg/debug_test.txt
Normal file
1
getpkg/debug_test.txt
Normal file
@ -0,0 +1 @@
|
||||
Debug content
|
@ -1,530 +1,270 @@
|
||||
#include "GetbinClient.hpp"
|
||||
#include <drogon/HttpClient.h>
|
||||
#include <trantor/net/EventLoop.h>
|
||||
#include <openssl/ssl.h>
|
||||
#include <openssl/opensslconf.h>
|
||||
#include <fstream>
|
||||
#include <sstream>
|
||||
#include <cpr/cpr.h>
|
||||
#include <nlohmann/json.hpp>
|
||||
#include <string>
|
||||
#include <fstream>
|
||||
#include <iostream>
|
||||
#include <thread>
|
||||
#include <chrono>
|
||||
#include <cstdio>
|
||||
#include <map>
|
||||
#include <atomic>
|
||||
#include <mutex>
|
||||
#include <condition_variable>
|
||||
#include <vector>
|
||||
#include <ctime>
|
||||
#include <algorithm>
|
||||
#include <filesystem>
|
||||
#include <sstream>
|
||||
#include <set>
|
||||
#include <algorithm>
|
||||
|
||||
using json = nlohmann::json;
|
||||
|
||||
static constexpr const char* SERVER_HOST = "getpkg.xyz";
|
||||
const std::string GetbinClient::SERVER_HOST = "getpkg.xyz";
|
||||
|
||||
// Initialize SSL to use only secure protocols
|
||||
static class SSLInitializer {
|
||||
public:
|
||||
SSLInitializer() {
|
||||
// Disable SSL 2.0, 3.0, TLS 1.0, and TLS 1.1
|
||||
SSL_load_error_strings();
|
||||
SSL_library_init();
|
||||
// Note: This doesn't completely silence the warning but ensures we're using secure protocols
|
||||
}
|
||||
} ssl_init;
|
||||
|
||||
static std::string find_ca_certificates() {
|
||||
// Common CA certificate locations across different Linux distributions
|
||||
const std::vector<std::string> ca_paths = {
|
||||
"/etc/ssl/certs/ca-certificates.crt", // Debian/Ubuntu/Raspbian
|
||||
"/etc/pki/tls/certs/ca-bundle.crt", // Fedora/RHEL/CentOS
|
||||
"/etc/ssl/ca-bundle.pem", // OpenSUSE
|
||||
"/etc/pki/tls/cert.pem", // Fedora/RHEL alternative
|
||||
"/etc/ssl/certs/ca-bundle.crt", // Some distros
|
||||
"/etc/ssl/cert.pem", // Alpine Linux
|
||||
"/usr/local/share/certs/ca-root-nss.crt", // FreeBSD
|
||||
"/etc/pki/ca-trust/extracted/pem/tls-ca-bundle.pem", // CentOS/RHEL 7+
|
||||
"/etc/ca-certificates/extracted/tls-ca-bundle.pem" // Arch Linux
|
||||
};
|
||||
|
||||
for (const auto& path : ca_paths) {
|
||||
std::ifstream file(path);
|
||||
if (file.good()) {
|
||||
file.close();
|
||||
return path;
|
||||
}
|
||||
}
|
||||
|
||||
return "";
|
||||
GetbinClient::GetbinClient() {
|
||||
// Initialize CPR (done automatically, but we could add global config here)
|
||||
}
|
||||
|
||||
GetbinClient::GetbinClient() {}
|
||||
std::string GetbinClient::getUserAgent() const {
|
||||
return "getpkg/1.0";
|
||||
}
|
||||
|
||||
bool GetbinClient::download(const std::string& toolName, const std::string& arch, const std::string& outPath) {
|
||||
bool success = false;
|
||||
bool done = false;
|
||||
std::mutex mtx;
|
||||
std::condition_variable cv;
|
||||
bool GetbinClient::download(const std::string& toolName, const std::string& arch, const std::string& outPath,
|
||||
ProgressCallback progressCallback) {
|
||||
try {
|
||||
std::string url = "https://" + SERVER_HOST + "/object/" + toolName + ":" + arch;
|
||||
|
||||
std::thread worker([&]() {
|
||||
trantor::EventLoop loop;
|
||||
cpr::Session session;
|
||||
session.SetUrl(cpr::Url{url});
|
||||
session.SetHeader(cpr::Header{{"User-Agent", getUserAgent()}});
|
||||
session.SetTimeout(cpr::Timeout{30000}); // 30 seconds
|
||||
session.SetVerifySsl(cpr::VerifySsl{true});
|
||||
|
||||
auto client = drogon::HttpClient::newHttpClient(
|
||||
"https://" + std::string(SERVER_HOST),
|
||||
&loop,
|
||||
false, // useOldTLS = false (disable TLS 1.0/1.1)
|
||||
true // validateCert = true
|
||||
);
|
||||
|
||||
// Configure SSL certificates for HTTPS
|
||||
std::string ca_path = find_ca_certificates();
|
||||
if (!ca_path.empty()) {
|
||||
// Use addSSLConfigs with proper parameter names for OpenSSL
|
||||
std::vector<std::pair<std::string, std::string>> sslConfigs;
|
||||
sslConfigs.push_back({"VerifyCAFile", ca_path});
|
||||
client->addSSLConfigs(sslConfigs);
|
||||
} else {
|
||||
// If no CA certificates found, print warning but continue
|
||||
std::cerr << "[GetbinClient] Warning: No system CA certificates found. SSL verification may fail." << std::endl;
|
||||
// Add progress callback if provided
|
||||
if (progressCallback) {
|
||||
session.SetProgressCallback(cpr::ProgressCallback{[progressCallback](cpr::cpr_off_t downloadTotal, cpr::cpr_off_t downloadNow,
|
||||
cpr::cpr_off_t uploadTotal, cpr::cpr_off_t uploadNow,
|
||||
intptr_t userdata) -> bool {
|
||||
return progressCallback(static_cast<size_t>(downloadNow), static_cast<size_t>(downloadTotal));
|
||||
}});
|
||||
}
|
||||
|
||||
client->enableCookies();
|
||||
client->setUserAgent("getpkg/1.0");
|
||||
auto response = session.Get();
|
||||
|
||||
std::string object_path = "/object/" + toolName + ":" + arch;
|
||||
|
||||
auto req = drogon::HttpRequest::newHttpRequest();
|
||||
req->setMethod(drogon::Get);
|
||||
req->setPath(object_path);
|
||||
|
||||
client->sendRequest(req, [&](drogon::ReqResult result, const drogon::HttpResponsePtr& response) {
|
||||
std::lock_guard<std::mutex> lock(mtx);
|
||||
if (result == drogon::ReqResult::Ok && response && response->getStatusCode() == drogon::k200OK) {
|
||||
if (response.status_code == 200) {
|
||||
std::ofstream ofs(outPath, std::ios::binary);
|
||||
if (ofs) {
|
||||
const auto& body = response->getBody();
|
||||
ofs.write(body.data(), body.size());
|
||||
success = ofs.good();
|
||||
ofs.write(response.text.data(), response.text.size());
|
||||
return ofs.good();
|
||||
}
|
||||
} else if (response.status_code == 404) {
|
||||
// Not found - this is expected for arch fallback
|
||||
return false;
|
||||
} else {
|
||||
std::cerr << "[GetbinClient::download] HTTP request failed." << std::endl;
|
||||
}
|
||||
done = true;
|
||||
cv.notify_one();
|
||||
loop.quit();
|
||||
}, 30.0); // 30 second timeout
|
||||
|
||||
loop.loop();
|
||||
});
|
||||
|
||||
// Wait for completion
|
||||
{
|
||||
std::unique_lock<std::mutex> lock(mtx);
|
||||
cv.wait(lock, [&] { return done; });
|
||||
std::cerr << "[GetbinClient::download] HTTP " << response.status_code << ": " << response.error.message << std::endl;
|
||||
}
|
||||
|
||||
worker.join();
|
||||
return success;
|
||||
}
|
||||
|
||||
bool GetbinClient::upload(const std::string& archivePath, std::string& outUrl, std::string& outHash, const std::string& token) {
|
||||
// Read file first
|
||||
std::ifstream ifs(archivePath, std::ios::binary);
|
||||
if (!ifs) {
|
||||
std::cerr << "[GetbinClient::upload] Failed to open archive file: " << archivePath << std::endl;
|
||||
return false;
|
||||
} catch (const std::exception& e) {
|
||||
std::cerr << "[GetbinClient::download] Exception: " << e.what() << std::endl;
|
||||
return false;
|
||||
}
|
||||
std::string file_content((std::istreambuf_iterator<char>(ifs)), std::istreambuf_iterator<char>());
|
||||
|
||||
// Compose metadata
|
||||
json metadata = { {"labeltags", json::array()} };
|
||||
std::string filename = archivePath.substr(archivePath.find_last_of("/\\") + 1);
|
||||
size_t dot = filename.find('.');
|
||||
std::string labeltag = dot != std::string::npos ? filename.substr(0, dot) : filename;
|
||||
metadata["labeltags"].push_back(labeltag);
|
||||
|
||||
bool success = false;
|
||||
bool done = false;
|
||||
std::mutex mtx;
|
||||
std::condition_variable cv;
|
||||
|
||||
std::thread worker([&]() {
|
||||
trantor::EventLoop loop;
|
||||
|
||||
auto client = drogon::HttpClient::newHttpClient(
|
||||
"https://" + std::string(SERVER_HOST),
|
||||
&loop,
|
||||
false, // useOldTLS = false (disable TLS 1.0/1.1)
|
||||
true // validateCert = true
|
||||
);
|
||||
|
||||
// Configure SSL certificates
|
||||
std::string ca_path = find_ca_certificates();
|
||||
std::vector<std::pair<std::string, std::string>> sslConfigs;
|
||||
if (!ca_path.empty()) {
|
||||
sslConfigs.push_back({"VerifyCAFile", ca_path});
|
||||
}
|
||||
// Configure SSL for secure connections
|
||||
client->addSSLConfigs(sslConfigs);
|
||||
|
||||
if (ca_path.empty()) {
|
||||
std::cerr << "[GetbinClient] Warning: No system CA certificates found. SSL verification may fail." << std::endl;
|
||||
}
|
||||
|
||||
client->enableCookies();
|
||||
client->setUserAgent("getpkg/1.0");
|
||||
|
||||
// Create upload file from memory content
|
||||
// First save content to a temporary file since UploadFile expects a file path
|
||||
std::string temp_file = "/tmp/getpkg_upload_" + std::to_string(std::time(nullptr)) + ".tgz";
|
||||
std::ofstream temp_ofs(temp_file, std::ios::binary);
|
||||
if (!temp_ofs) {
|
||||
std::cerr << "[GetbinClient::upload] Failed to create temporary file: " << temp_file << std::endl;
|
||||
success = false;
|
||||
done = true;
|
||||
cv.notify_one();
|
||||
loop.quit();
|
||||
return;
|
||||
}
|
||||
temp_ofs.write(file_content.data(), file_content.size());
|
||||
temp_ofs.close();
|
||||
|
||||
// Create upload request with file
|
||||
drogon::UploadFile upload_file(temp_file);
|
||||
|
||||
auto req = drogon::HttpRequest::newFileUploadRequest({upload_file});
|
||||
req->setMethod(drogon::Put);
|
||||
req->setPath("/upload");
|
||||
req->addHeader("Authorization", "Bearer " + token);
|
||||
|
||||
// Add metadata as form parameter
|
||||
req->setParameter("metadata", metadata.dump());
|
||||
|
||||
client->sendRequest(req, [&](drogon::ReqResult result, const drogon::HttpResponsePtr& response) {
|
||||
std::lock_guard<std::mutex> lock(mtx);
|
||||
if (result == drogon::ReqResult::Ok && response) {
|
||||
int status_code = static_cast<int>(response->getStatusCode());
|
||||
std::string response_body(response->getBody());
|
||||
|
||||
if (status_code == 200 || status_code == 201) {
|
||||
try {
|
||||
auto resp_json = json::parse(response_body);
|
||||
if (resp_json.contains("url")) outUrl = resp_json["url"].get<std::string>();
|
||||
if (resp_json.contains("hash")) outHash = resp_json["hash"].get<std::string>();
|
||||
success = true;
|
||||
} catch (const std::exception& e) {
|
||||
std::cerr << "[GetbinClient::upload] Failed to parse JSON response: " << e.what() << std::endl;
|
||||
std::cerr << "[GetbinClient::upload] Response body: " << response_body << std::endl;
|
||||
}
|
||||
} else {
|
||||
std::cerr << "[GetbinClient::upload] HTTP error: status code " << status_code << std::endl;
|
||||
std::cerr << "[GetbinClient::upload] Response body: " << response_body << std::endl;
|
||||
}
|
||||
} else {
|
||||
std::cerr << "[GetbinClient::upload] HTTP /upload request failed." << std::endl;
|
||||
}
|
||||
done = true;
|
||||
cv.notify_one();
|
||||
loop.quit();
|
||||
}, 60.0); // 60 second timeout
|
||||
|
||||
loop.loop();
|
||||
|
||||
// Clean up temporary file
|
||||
std::remove(temp_file.c_str());
|
||||
});
|
||||
|
||||
// Wait for completion
|
||||
{
|
||||
std::unique_lock<std::mutex> lock(mtx);
|
||||
cv.wait(lock, [&] { return done; });
|
||||
}
|
||||
|
||||
worker.join();
|
||||
return success;
|
||||
}
|
||||
|
||||
bool GetbinClient::getHash(const std::string& toolName, const std::string& arch, std::string& outHash) {
|
||||
bool success = false;
|
||||
bool done = false;
|
||||
std::mutex mtx;
|
||||
std::condition_variable cv;
|
||||
|
||||
std::thread worker([&]() {
|
||||
trantor::EventLoop loop;
|
||||
|
||||
auto client = drogon::HttpClient::newHttpClient(
|
||||
"https://" + std::string(SERVER_HOST),
|
||||
&loop,
|
||||
false, // useOldTLS = false (disable TLS 1.0/1.1)
|
||||
true // validateCert = true
|
||||
);
|
||||
|
||||
// Configure SSL certificates
|
||||
std::string ca_path = find_ca_certificates();
|
||||
std::vector<std::pair<std::string, std::string>> sslConfigs;
|
||||
if (!ca_path.empty()) {
|
||||
sslConfigs.push_back({"VerifyCAFile", ca_path});
|
||||
}
|
||||
// Configure SSL for secure connections
|
||||
client->addSSLConfigs(sslConfigs);
|
||||
|
||||
if (ca_path.empty()) {
|
||||
std::cerr << "[GetbinClient] Warning: No system CA certificates found. SSL verification may fail." << std::endl;
|
||||
}
|
||||
|
||||
client->enableCookies();
|
||||
client->setUserAgent("getpkg/1.0");
|
||||
|
||||
std::string hash_path = "/hash/" + toolName + ":" + arch;
|
||||
|
||||
auto req = drogon::HttpRequest::newHttpRequest();
|
||||
req->setMethod(drogon::Get);
|
||||
req->setPath(hash_path);
|
||||
|
||||
client->sendRequest(req, [&](drogon::ReqResult result, const drogon::HttpResponsePtr& response) {
|
||||
std::lock_guard<std::mutex> lock(mtx);
|
||||
if (result == drogon::ReqResult::Ok && response && response->getStatusCode() == drogon::k200OK) {
|
||||
std::string response_body(response->getBody());
|
||||
|
||||
// Try to parse hash from response body
|
||||
bool GetbinClient::upload(const std::string& archivePath, std::string& outUrl, std::string& outHash,
|
||||
const std::string& token, ProgressCallback progressCallback) {
|
||||
try {
|
||||
// Try JSON first
|
||||
auto resp_json = json::parse(response_body);
|
||||
if (resp_json.contains("hash")) {
|
||||
outHash = resp_json["hash"].get<std::string>();
|
||||
success = true;
|
||||
std::string url = "https://" + SERVER_HOST + "/upload";
|
||||
|
||||
cpr::Session session;
|
||||
session.SetUrl(cpr::Url{url});
|
||||
session.SetHeader(cpr::Header{
|
||||
{"User-Agent", getUserAgent()},
|
||||
{"Authorization", "Bearer " + token}
|
||||
});
|
||||
session.SetTimeout(cpr::Timeout{300000}); // 5 minutes for uploads
|
||||
session.SetVerifySsl(cpr::VerifySsl{true});
|
||||
|
||||
|
||||
// Extract tool name and arch from archive path for labeltags
|
||||
// Archive path format: /path/to/tool-name:arch.tgz or similar
|
||||
std::string archiveName = std::filesystem::path(archivePath).filename().string();
|
||||
std::string toolNameArch = archiveName;
|
||||
if (toolNameArch.ends_with(".tgz")) {
|
||||
toolNameArch = toolNameArch.substr(0, toolNameArch.length() - 4);
|
||||
}
|
||||
} catch (...) {
|
||||
// Not JSON, treat as plain text
|
||||
outHash = response_body;
|
||||
|
||||
// Create metadata JSON with labeltags
|
||||
json metadata;
|
||||
metadata["labeltags"] = json::array({toolNameArch});
|
||||
|
||||
// Set up multipart form with file and metadata
|
||||
session.SetMultipart(cpr::Multipart{
|
||||
cpr::Part{"file", cpr::File{archivePath}},
|
||||
cpr::Part{"metadata", metadata.dump(), "application/json"}
|
||||
});
|
||||
|
||||
// Add progress callback if provided
|
||||
if (progressCallback) {
|
||||
session.SetProgressCallback(cpr::ProgressCallback{[progressCallback](cpr::cpr_off_t downloadTotal, cpr::cpr_off_t downloadNow,
|
||||
cpr::cpr_off_t uploadTotal, cpr::cpr_off_t uploadNow,
|
||||
intptr_t userdata) -> bool {
|
||||
return progressCallback(static_cast<size_t>(uploadNow), static_cast<size_t>(uploadTotal));
|
||||
}});
|
||||
}
|
||||
|
||||
auto response = session.Put();
|
||||
|
||||
if (response.status_code == 200) {
|
||||
try {
|
||||
auto resp_json = json::parse(response.text);
|
||||
if (resp_json.contains("hash") && resp_json.contains("result") && resp_json["result"] == "success") {
|
||||
outUrl = "https://" + SERVER_HOST + "/object/" + resp_json["hash"].get<std::string>();
|
||||
outHash = resp_json["hash"].get<std::string>();
|
||||
return true;
|
||||
}
|
||||
} catch (const json::exception& e) {
|
||||
// Try to extract from plain text response
|
||||
outUrl = "";
|
||||
outHash = response.text;
|
||||
// Remove trailing newline if present
|
||||
if (!outHash.empty() && outHash.back() == '\n') {
|
||||
outHash.pop_back();
|
||||
}
|
||||
success = !outHash.empty();
|
||||
return !outHash.empty();
|
||||
}
|
||||
} else {
|
||||
std::cerr << "[GetbinClient::upload] HTTP " << response.status_code << ": " << response.error.message << std::endl;
|
||||
if (!response.text.empty()) {
|
||||
std::cerr << "[GetbinClient::upload] Response: " << response.text << std::endl;
|
||||
}
|
||||
}
|
||||
done = true;
|
||||
cv.notify_one();
|
||||
loop.quit();
|
||||
}, 10.0); // 10 second timeout
|
||||
|
||||
loop.loop();
|
||||
});
|
||||
return false;
|
||||
} catch (const std::exception& e) {
|
||||
std::cerr << "[GetbinClient::upload] Exception: " << e.what() << std::endl;
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
// Wait for completion
|
||||
{
|
||||
std::unique_lock<std::mutex> lock(mtx);
|
||||
cv.wait(lock, [&] { return done; });
|
||||
bool GetbinClient::getHash(const std::string& toolName, const std::string& arch, std::string& outHash) {
|
||||
try {
|
||||
std::string url = "https://" + SERVER_HOST + "/hash/" + toolName + ":" + arch;
|
||||
|
||||
auto response = cpr::Get(cpr::Url{url},
|
||||
cpr::Header{{"User-Agent", getUserAgent()}},
|
||||
cpr::Timeout{10000}, // 10 seconds
|
||||
cpr::VerifySsl{true});
|
||||
|
||||
if (response.status_code == 200) {
|
||||
try {
|
||||
// Try JSON first
|
||||
auto resp_json = json::parse(response.text);
|
||||
if (resp_json.contains("hash")) {
|
||||
outHash = resp_json["hash"].get<std::string>();
|
||||
return true;
|
||||
}
|
||||
} catch (const json::exception&) {
|
||||
// Not JSON, treat as plain text
|
||||
outHash = response.text;
|
||||
// Remove trailing newline if present
|
||||
if (!outHash.empty() && outHash.back() == '\n') {
|
||||
outHash.pop_back();
|
||||
}
|
||||
return !outHash.empty();
|
||||
}
|
||||
} else if (response.status_code == 404) {
|
||||
// Not found - this is expected for non-existent tools/archs
|
||||
return false;
|
||||
} else {
|
||||
std::cerr << "[GetbinClient::getHash] HTTP " << response.status_code << ": " << response.error.message << std::endl;
|
||||
}
|
||||
|
||||
worker.join();
|
||||
return success;
|
||||
return false;
|
||||
} catch (const std::exception& e) {
|
||||
std::cerr << "[GetbinClient::getHash] Exception: " << e.what() << std::endl;
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
bool GetbinClient::deleteObject(const std::string& hash, const std::string& token) {
|
||||
bool success = false;
|
||||
bool done = false;
|
||||
std::mutex mtx;
|
||||
std::condition_variable cv;
|
||||
|
||||
std::thread worker([&]() {
|
||||
trantor::EventLoop loop;
|
||||
|
||||
auto client = drogon::HttpClient::newHttpClient(
|
||||
"https://" + std::string(SERVER_HOST),
|
||||
&loop,
|
||||
false, // useOldTLS = false (disable TLS 1.0/1.1)
|
||||
true // validateCert = true
|
||||
);
|
||||
|
||||
// Configure SSL certificates
|
||||
std::string ca_path = find_ca_certificates();
|
||||
std::vector<std::pair<std::string, std::string>> sslConfigs;
|
||||
if (!ca_path.empty()) {
|
||||
sslConfigs.push_back({"VerifyCAFile", ca_path});
|
||||
}
|
||||
// Configure SSL for secure connections
|
||||
client->addSSLConfigs(sslConfigs);
|
||||
|
||||
if (ca_path.empty()) {
|
||||
std::cerr << "[GetbinClient] Warning: No system CA certificates found. SSL verification may fail." << std::endl;
|
||||
}
|
||||
|
||||
client->enableCookies();
|
||||
client->setUserAgent("getpkg/1.0");
|
||||
|
||||
std::string delete_path = "/deleteobject?hash=" + hash;
|
||||
|
||||
auto req = drogon::HttpRequest::newHttpRequest();
|
||||
req->setMethod(drogon::Get);
|
||||
req->setPath(delete_path);
|
||||
req->addHeader("Authorization", "Bearer " + token);
|
||||
|
||||
client->sendRequest(req, [&](drogon::ReqResult result, const drogon::HttpResponsePtr& response) {
|
||||
std::lock_guard<std::mutex> lock(mtx);
|
||||
if (result == drogon::ReqResult::Ok && response) {
|
||||
int status_code = static_cast<int>(response->getStatusCode());
|
||||
std::string response_body(response->getBody());
|
||||
|
||||
if (status_code == 200) {
|
||||
// Check if the response indicates success
|
||||
try {
|
||||
auto resp_json = json::parse(response_body);
|
||||
if (resp_json.contains("result") && resp_json["result"] == "success") {
|
||||
success = true;
|
||||
}
|
||||
} catch (...) {
|
||||
// If not JSON, assume success if 200 OK
|
||||
success = true;
|
||||
}
|
||||
std::string url = "https://" + SERVER_HOST + "/deleteobject?hash=" + hash;
|
||||
|
||||
auto response = cpr::Get(cpr::Url{url},
|
||||
cpr::Header{
|
||||
{"User-Agent", getUserAgent()},
|
||||
{"Authorization", "Bearer " + token}
|
||||
},
|
||||
cpr::Timeout{30000}, // 30 seconds
|
||||
cpr::VerifySsl{true});
|
||||
|
||||
if (response.status_code == 200) {
|
||||
return true;
|
||||
} else {
|
||||
std::cerr << "[GetbinClient::deleteObject] HTTP error: status code " << status_code << std::endl;
|
||||
std::cerr << "[GetbinClient::deleteObject] Response body: " << response_body << std::endl;
|
||||
std::cerr << "[GetbinClient::deleteObject] HTTP " << response.status_code << ": " << response.error.message << std::endl;
|
||||
if (!response.text.empty()) {
|
||||
std::cerr << "[GetbinClient::deleteObject] Response: " << response.text << std::endl;
|
||||
}
|
||||
} else {
|
||||
std::cerr << "[GetbinClient::deleteObject] HTTP request failed." << std::endl;
|
||||
}
|
||||
done = true;
|
||||
cv.notify_one();
|
||||
loop.quit();
|
||||
}, 10.0); // 10 second timeout
|
||||
|
||||
loop.loop();
|
||||
});
|
||||
|
||||
// Wait for completion
|
||||
{
|
||||
std::unique_lock<std::mutex> lock(mtx);
|
||||
cv.wait(lock, [&] { return done; });
|
||||
}
|
||||
|
||||
worker.join();
|
||||
return success;
|
||||
return false;
|
||||
} catch (const std::exception& e) {
|
||||
std::cerr << "[GetbinClient::deleteObject] Exception: " << e.what() << std::endl;
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
bool GetbinClient::listPackages(std::vector<std::string>& outPackages) {
|
||||
outPackages.clear();
|
||||
|
||||
// Set up SSL configuration
|
||||
std::string ca_path = find_ca_certificates();
|
||||
|
||||
bool success = false;
|
||||
bool done = false;
|
||||
std::mutex mtx;
|
||||
std::condition_variable cv;
|
||||
|
||||
std::thread worker([&]() {
|
||||
trantor::EventLoop loop;
|
||||
|
||||
auto client = drogon::HttpClient::newHttpClient(
|
||||
"https://" + std::string(SERVER_HOST),
|
||||
&loop,
|
||||
false, // useOldTLS = false (disable TLS 1.0/1.1)
|
||||
true // validateCert = true
|
||||
);
|
||||
std::vector<std::pair<std::string, std::string>> sslConfigs;
|
||||
if (!ca_path.empty()) {
|
||||
sslConfigs.push_back({"VerifyCAFile", ca_path});
|
||||
}
|
||||
// Configure SSL for secure connections
|
||||
client->addSSLConfigs(sslConfigs);
|
||||
|
||||
auto req = drogon::HttpRequest::newHttpRequest();
|
||||
req->setMethod(drogon::Get);
|
||||
req->setPath("/dir");
|
||||
|
||||
client->sendRequest(req, [&](drogon::ReqResult result, const drogon::HttpResponsePtr& response) {
|
||||
if (result == drogon::ReqResult::Ok) {
|
||||
int status_code = response->getStatusCode();
|
||||
std::string response_body = std::string(response->getBody());
|
||||
|
||||
if (status_code == 200) {
|
||||
try {
|
||||
json json_response = json::parse(response_body);
|
||||
std::string url = "https://" + SERVER_HOST + "/dir";
|
||||
|
||||
if (json_response.contains("entries") && json_response["entries"].is_array()) {
|
||||
for (const auto& entry : json_response["entries"]) {
|
||||
auto response = cpr::Get(cpr::Url{url},
|
||||
cpr::Header{{"User-Agent", getUserAgent()}},
|
||||
cpr::Timeout{30000}, // 30 seconds
|
||||
cpr::VerifySsl{true});
|
||||
|
||||
if (response.status_code == 200) {
|
||||
try {
|
||||
auto resp_json = json::parse(response.text);
|
||||
if (resp_json.contains("entries") && resp_json["entries"].is_array()) {
|
||||
outPackages.clear();
|
||||
std::set<std::string> uniqueTools;
|
||||
|
||||
for (const auto& entry : resp_json["entries"]) {
|
||||
if (entry.contains("labeltags") && entry["labeltags"].is_array()) {
|
||||
for (const auto& labeltag : entry["labeltags"]) {
|
||||
if (labeltag.is_string()) {
|
||||
std::string name = labeltag.get<std::string>();
|
||||
// Extract tool name (remove architecture suffix if present)
|
||||
size_t colon_pos = name.find(":");
|
||||
if (colon_pos != std::string::npos) {
|
||||
name = name.substr(0, colon_pos);
|
||||
std::string tag = labeltag.get<std::string>();
|
||||
// Extract tool name from "tool:arch" format
|
||||
size_t colonPos = tag.find(":");
|
||||
if (colonPos != std::string::npos) {
|
||||
std::string toolName = tag.substr(0, colonPos);
|
||||
if (!toolName.empty()) {
|
||||
uniqueTools.insert(toolName);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Convert set to vector
|
||||
for (const auto& tool : uniqueTools) {
|
||||
outPackages.push_back(tool);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
} catch (const json::exception&) {
|
||||
// Try to parse as newline-separated list
|
||||
outPackages.clear();
|
||||
std::istringstream stream(response.text);
|
||||
std::string line;
|
||||
while (std::getline(stream, line)) {
|
||||
if (!line.empty()) {
|
||||
outPackages.push_back(line);
|
||||
}
|
||||
}
|
||||
return !outPackages.empty();
|
||||
}
|
||||
} else {
|
||||
std::cerr << "[GetbinClient::listPackages] HTTP " << response.status_code << ": " << response.error.message << std::endl;
|
||||
}
|
||||
|
||||
// Skip empty names
|
||||
if (name.empty()) continue;
|
||||
|
||||
// Add to list if not already present
|
||||
if (std::find(outPackages.begin(), outPackages.end(), name) == outPackages.end()) {
|
||||
outPackages.push_back(name);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
success = true;
|
||||
}
|
||||
return false;
|
||||
} catch (const std::exception& e) {
|
||||
std::cerr << "[GetbinClient::listPackages] JSON parse error: " << e.what() << std::endl;
|
||||
std::cerr << "[GetbinClient::listPackages] Exception: " << e.what() << std::endl;
|
||||
return false;
|
||||
}
|
||||
} else {
|
||||
std::cerr << "[GetbinClient::listPackages] HTTP error: status code " << status_code << std::endl;
|
||||
}
|
||||
} else {
|
||||
std::cerr << "[GetbinClient::listPackages] HTTP request failed." << std::endl;
|
||||
}
|
||||
done = true;
|
||||
cv.notify_one();
|
||||
loop.quit();
|
||||
}, 10.0);
|
||||
|
||||
loop.loop();
|
||||
});
|
||||
|
||||
// Wait for completion
|
||||
{
|
||||
std::unique_lock<std::mutex> lock(mtx);
|
||||
cv.wait(lock, [&] { return done; });
|
||||
}
|
||||
|
||||
worker.join();
|
||||
|
||||
// Filter out duplicates where we have both toolname and toolname-noarch
|
||||
// Keep the base name and remove the -noarch variant
|
||||
std::vector<std::string> filteredPackages;
|
||||
std::set<std::string> baseNames;
|
||||
|
||||
// First pass: collect all base names (without -noarch)
|
||||
for (const auto& pkg : outPackages) {
|
||||
const std::string suffix = "-noarch";
|
||||
if (pkg.length() < suffix.length() || pkg.substr(pkg.length() - suffix.length()) != suffix) {
|
||||
baseNames.insert(pkg);
|
||||
}
|
||||
}
|
||||
|
||||
// Second pass: add packages, skipping -noarch variants if base exists
|
||||
for (const auto& pkg : outPackages) {
|
||||
const std::string suffix = "-noarch";
|
||||
if (pkg.length() >= suffix.length() && pkg.substr(pkg.length() - suffix.length()) == suffix) {
|
||||
std::string baseName = pkg.substr(0, pkg.length() - suffix.length());
|
||||
if (baseNames.find(baseName) == baseNames.end()) {
|
||||
filteredPackages.push_back(pkg); // Keep -noarch only if no base version
|
||||
}
|
||||
} else {
|
||||
filteredPackages.push_back(pkg); // Always keep base versions
|
||||
}
|
||||
}
|
||||
|
||||
outPackages = std::move(filteredPackages);
|
||||
|
||||
// Sort the packages for better display
|
||||
std::sort(outPackages.begin(), outPackages.end());
|
||||
|
||||
return success;
|
||||
}
|
@ -1,13 +1,24 @@
|
||||
#pragma once
|
||||
#include <string>
|
||||
#include <vector>
|
||||
#include <functional>
|
||||
|
||||
class GetbinClient {
|
||||
public:
|
||||
GetbinClient();
|
||||
bool download(const std::string& toolName, const std::string& arch, const std::string& outPath);
|
||||
bool upload(const std::string& archivePath, std::string& outUrl, std::string& outHash, const std::string& token);
|
||||
|
||||
// Progress callback: (downloaded_bytes, total_bytes) -> should_continue
|
||||
using ProgressCallback = std::function<bool(size_t, size_t)>;
|
||||
|
||||
bool download(const std::string& toolName, const std::string& arch, const std::string& outPath,
|
||||
ProgressCallback progressCallback = nullptr);
|
||||
bool upload(const std::string& archivePath, std::string& outUrl, std::string& outHash, const std::string& token,
|
||||
ProgressCallback progressCallback = nullptr);
|
||||
bool getHash(const std::string& toolName, const std::string& arch, std::string& outHash);
|
||||
bool deleteObject(const std::string& hash, const std::string& token);
|
||||
bool listPackages(std::vector<std::string>& outPackages);
|
||||
|
||||
private:
|
||||
static const std::string SERVER_HOST;
|
||||
std::string getUserAgent() const;
|
||||
};
|
@ -200,27 +200,43 @@ int install_tool(int argc, char* argv[]) {
|
||||
// Download tool - try arch-specific version first, then universal fallback
|
||||
GetbinClient getbin2;
|
||||
std::string downloadArch = arch;
|
||||
//std::cout << "Downloading " << toolName << ":" << arch << "..." << std::endl;
|
||||
if (!getbin2.download(toolName, arch, archivePath.string())) {
|
||||
|
||||
// Progress callback for downloads
|
||||
auto progressCallback = [&toolName](size_t downloaded, size_t total) -> bool {
|
||||
if (total > 0) {
|
||||
int percent = (downloaded * 100) / total;
|
||||
std::cout << "\rDownloading " << toolName << "... " << percent << "%" << std::flush;
|
||||
} else {
|
||||
std::cout << "\rDownloading " << toolName << "... " << downloaded << " bytes" << std::flush;
|
||||
}
|
||||
return true; // Continue download
|
||||
};
|
||||
|
||||
std::cout << "Downloading " << toolName << "..." << std::flush;
|
||||
if (!getbin2.download(toolName, arch, archivePath.string(), progressCallback)) {
|
||||
// Try universal version as fallback
|
||||
//std::cout << "Arch-specific version not found, trying universal version..." << std::endl;
|
||||
//std::cout << "Downloading " << toolName << ":universal..." << std::endl;
|
||||
if (!getbin2.download(toolName, "universal", archivePath.string())) {
|
||||
std::cerr << "Failed to download tool archive (tried both " << arch << " and universal)." << std::endl;
|
||||
std::cout << "\rArch-specific version not found, trying universal..." << std::endl;
|
||||
if (!getbin2.download(toolName, "universal", archivePath.string(), progressCallback)) {
|
||||
std::cerr << "\rFailed to download tool archive (tried both " << arch << " and universal)." << std::endl;
|
||||
return 1;
|
||||
}
|
||||
downloadArch = "universal";
|
||||
}
|
||||
std::cout << "\rDownloading " << toolName << "... done" << std::endl;
|
||||
|
||||
// Unpack tool
|
||||
std::cout << "Unpacking..." << std::flush;
|
||||
if (!common::unpack_tgz(archivePath.string(), binDir.string())) {
|
||||
std::cerr << "Failed to unpack tool archive." << std::endl;
|
||||
std::cerr << "\rFailed to unpack tool archive." << std::endl;
|
||||
return 1;
|
||||
}
|
||||
std::cout << "\rUnpacking... done" << std::endl;
|
||||
|
||||
// Add to PATH and autocomplete
|
||||
std::cout << "Configuring..." << std::flush;
|
||||
scriptManager.addToolEntry(toolName, binDir.string());
|
||||
scriptManager.addAutocomplete(toolName);
|
||||
std::cout << "\rConfiguring... done" << std::endl;
|
||||
|
||||
// Get tool info
|
||||
std::string hash;
|
||||
@ -314,10 +330,24 @@ int publish_tool(int argc, char* argv[]) {
|
||||
}
|
||||
GetbinClient getbin;
|
||||
std::string url, hash;
|
||||
if (!getbin.upload(archivePath.string(), url, hash, token)) {
|
||||
std::cerr << "Failed to upload archive." << std::endl;
|
||||
|
||||
// Progress callback for upload
|
||||
auto uploadProgressCallback = [](size_t uploaded, size_t total) -> bool {
|
||||
if (total > 0) {
|
||||
int percent = (uploaded * 100) / total;
|
||||
std::cout << "\rUploading... " << percent << "%" << std::flush;
|
||||
} else {
|
||||
std::cout << "\rUploading... " << uploaded << " bytes" << std::flush;
|
||||
}
|
||||
return true; // Continue upload
|
||||
};
|
||||
|
||||
std::cout << "Uploading..." << std::flush;
|
||||
if (!getbin.upload(archivePath.string(), url, hash, token, uploadProgressCallback)) {
|
||||
std::cerr << "\rFailed to upload archive." << std::endl;
|
||||
return 1;
|
||||
}
|
||||
std::cout << "\rUploading... done" << std::endl;
|
||||
std::cout << "Published! URL: " << url << "\nHash: " << hash << std::endl;
|
||||
return 0;
|
||||
}
|
||||
@ -326,73 +356,161 @@ int update_tool(int argc, char* argv[]) {
|
||||
std::string home = get_home();
|
||||
std::filesystem::path configDir = std::filesystem::path(home) / ".config/getpkg";
|
||||
|
||||
// Collect all installed tools
|
||||
std::vector<std::tuple<std::string, std::string, std::string>> updateResults; // name, status, version
|
||||
|
||||
// Capture stdout to process install_tool output
|
||||
auto processToolUpdate = [&](const std::string& toolName) -> std::tuple<std::string, std::string> {
|
||||
// Redirect stdout and stderr to capture output
|
||||
std::stringstream buffer;
|
||||
std::stringstream errBuffer;
|
||||
std::streambuf* oldOut = std::cout.rdbuf(buffer.rdbuf());
|
||||
std::streambuf* oldErr = std::cerr.rdbuf(errBuffer.rdbuf());
|
||||
|
||||
char* toolArgv[] = {argv[0], (char*)"install", (char*)toolName.c_str()};
|
||||
int result = install_tool(3, toolArgv);
|
||||
|
||||
// Restore stdout and stderr
|
||||
std::cout.rdbuf(oldOut);
|
||||
std::cerr.rdbuf(oldErr);
|
||||
|
||||
std::string output = buffer.str();
|
||||
std::string status = "Failed";
|
||||
std::string version = "-";
|
||||
|
||||
if (result == 0) {
|
||||
if (output.find("is already up to date") != std::string::npos) {
|
||||
status = "Up to date";
|
||||
} else if (output.find("Installed " + toolName + " successfully") != std::string::npos) {
|
||||
// Check if it was an update or fresh install
|
||||
if (output.find("Updating " + toolName) != std::string::npos) {
|
||||
status = "Updated";
|
||||
} else {
|
||||
status = "Installed";
|
||||
}
|
||||
}
|
||||
|
||||
// Try to get version from config
|
||||
std::filesystem::path toolInfoPath = configDir / (toolName + ".json");
|
||||
if (std::filesystem::exists(toolInfoPath)) {
|
||||
std::ifstream tfile(toolInfoPath);
|
||||
json toolInfo;
|
||||
tfile >> toolInfo;
|
||||
version = toolInfo.value("version", "-");
|
||||
if (!version.empty() && version.back() == '\n') version.pop_back();
|
||||
// If version is empty, try to show something useful
|
||||
if (version.empty() || version == "-") {
|
||||
version = "installed";
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return std::make_tuple(status, version);
|
||||
// Structure to hold tool information
|
||||
struct ToolInfo {
|
||||
std::string name;
|
||||
std::string localHash;
|
||||
std::string remoteHash;
|
||||
std::string arch;
|
||||
std::string version;
|
||||
bool needsUpdate = false;
|
||||
std::string status = "Up to date";
|
||||
};
|
||||
|
||||
// First update getpkg itself
|
||||
auto [getpkgStatus, getpkgVersion] = processToolUpdate("getpkg");
|
||||
updateResults.push_back(std::make_tuple("getpkg", getpkgStatus, getpkgVersion));
|
||||
std::vector<ToolInfo> tools;
|
||||
|
||||
// Then update all other installed tools
|
||||
// Collect all installed tools
|
||||
if (std::filesystem::exists(configDir)) {
|
||||
for (const auto& entry : std::filesystem::directory_iterator(configDir)) {
|
||||
if (entry.path().extension() == ".json") {
|
||||
std::string tname = entry.path().stem();
|
||||
if (tname != "getpkg") { // Skip getpkg since we already did it
|
||||
auto [status, version] = processToolUpdate(tname);
|
||||
updateResults.push_back(std::make_tuple(tname, status, version));
|
||||
|
||||
ToolInfo tool;
|
||||
tool.name = tname;
|
||||
|
||||
// Read local tool info
|
||||
std::ifstream tfile(entry.path());
|
||||
if (tfile.good()) {
|
||||
json toolInfo;
|
||||
tfile >> toolInfo;
|
||||
tool.localHash = toolInfo.value("hash", "");
|
||||
tool.arch = toolInfo.value("arch", get_arch());
|
||||
tool.version = toolInfo.value("version", "-");
|
||||
if (!tool.version.empty() && tool.version.back() == '\n') {
|
||||
tool.version.pop_back();
|
||||
}
|
||||
if (tool.version.empty() || tool.version == "-") {
|
||||
tool.version = "installed";
|
||||
}
|
||||
}
|
||||
|
||||
tools.push_back(tool);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (tools.empty()) {
|
||||
std::cout << "No tools installed." << std::endl;
|
||||
return 0;
|
||||
}
|
||||
|
||||
// Step 1: Check for updates (with progress)
|
||||
std::cout << "Checking " << tools.size() << " tools for updates..." << std::endl;
|
||||
|
||||
GetbinClient getbin;
|
||||
for (size_t i = 0; i < tools.size(); ++i) {
|
||||
auto& tool = tools[i];
|
||||
|
||||
// Show progress
|
||||
std::cout << "\r[" << (i + 1) << "/" << tools.size() << "] Checking " << tool.name << "..." << std::flush;
|
||||
|
||||
// Check remote hash
|
||||
std::string remoteHash;
|
||||
if (getbin.getHash(tool.name, tool.arch, remoteHash) && !remoteHash.empty()) {
|
||||
tool.remoteHash = remoteHash;
|
||||
if (tool.localHash != remoteHash) {
|
||||
tool.needsUpdate = true;
|
||||
tool.status = "Needs update";
|
||||
}
|
||||
} else {
|
||||
tool.status = "Check failed";
|
||||
}
|
||||
}
|
||||
std::cout << "\r" << std::string(50, ' ') << "\r" << std::flush; // Clear progress line
|
||||
|
||||
// Step 2: Update tools that need updating
|
||||
std::vector<std::tuple<std::string, std::string, std::string>> updateResults;
|
||||
|
||||
// First update getpkg if it needs updating
|
||||
auto getpkgIt = std::find_if(tools.begin(), tools.end(),
|
||||
[](const ToolInfo& t) { return t.name == "getpkg"; });
|
||||
|
||||
if (getpkgIt != tools.end() && getpkgIt->needsUpdate) {
|
||||
std::cout << "Updating getpkg..." << std::flush;
|
||||
|
||||
// Use install_tool for actual update
|
||||
std::stringstream buffer, errBuffer;
|
||||
std::streambuf* oldOut = std::cout.rdbuf(buffer.rdbuf());
|
||||
std::streambuf* oldErr = std::cerr.rdbuf(errBuffer.rdbuf());
|
||||
|
||||
char* toolArgv[] = {argv[0], (char*)"install", (char*)"getpkg"};
|
||||
int result = install_tool(3, toolArgv);
|
||||
|
||||
std::cout.rdbuf(oldOut);
|
||||
std::cerr.rdbuf(oldErr);
|
||||
|
||||
if (result == 0) {
|
||||
getpkgIt->status = "Updated";
|
||||
std::cout << " Updated" << std::endl;
|
||||
} else {
|
||||
getpkgIt->status = "Failed";
|
||||
std::cout << " Failed" << std::endl;
|
||||
}
|
||||
}
|
||||
|
||||
// Update other tools
|
||||
int toolsToUpdate = std::count_if(tools.begin(), tools.end(),
|
||||
[](const ToolInfo& t) { return t.needsUpdate && t.name != "getpkg"; });
|
||||
|
||||
if (toolsToUpdate > 0) {
|
||||
std::cout << "Updating " << toolsToUpdate << " tools..." << std::endl;
|
||||
|
||||
int updatedCount = 0;
|
||||
for (auto& tool : tools) {
|
||||
if (tool.needsUpdate && tool.name != "getpkg") {
|
||||
updatedCount++;
|
||||
std::cout << "[" << updatedCount << "/" << toolsToUpdate << "] Updating " << tool.name << "..." << std::flush;
|
||||
|
||||
// Use install_tool for actual update
|
||||
std::stringstream buffer, errBuffer;
|
||||
std::streambuf* oldOut = std::cout.rdbuf(buffer.rdbuf());
|
||||
std::streambuf* oldErr = std::cerr.rdbuf(errBuffer.rdbuf());
|
||||
|
||||
char* toolArgv[] = {argv[0], (char*)"install", (char*)tool.name.c_str()};
|
||||
int result = install_tool(3, toolArgv);
|
||||
|
||||
std::cout.rdbuf(oldOut);
|
||||
std::cerr.rdbuf(oldErr);
|
||||
|
||||
if (result == 0) {
|
||||
tool.status = "Updated";
|
||||
std::cout << " Updated" << std::endl;
|
||||
|
||||
// Re-read version after update
|
||||
std::filesystem::path toolInfoPath = configDir / (tool.name + ".json");
|
||||
if (std::filesystem::exists(toolInfoPath)) {
|
||||
std::ifstream tfile(toolInfoPath);
|
||||
json toolInfo;
|
||||
tfile >> toolInfo;
|
||||
tool.version = toolInfo.value("version", tool.version);
|
||||
if (!tool.version.empty() && tool.version.back() == '\n') {
|
||||
tool.version.pop_back();
|
||||
}
|
||||
if (tool.version.empty() || tool.version == "-") {
|
||||
tool.version = "installed";
|
||||
}
|
||||
}
|
||||
} else {
|
||||
tool.status = "Failed";
|
||||
std::cout << " Failed" << std::endl;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Prepare results for display
|
||||
for (const auto& tool : tools) {
|
||||
updateResults.push_back(std::make_tuple(tool.name, tool.status, tool.version));
|
||||
}
|
||||
|
||||
// Display results in a table
|
||||
@ -706,7 +824,7 @@ int list_packages(int argc, char* argv[]) {
|
||||
for (const auto& packageName : availablePackages) {
|
||||
std::string status = "Available";
|
||||
std::string localVersion = "-";
|
||||
std::string remoteStatus = "✓";
|
||||
std::string remoteStatus = "-";
|
||||
|
||||
auto it = installedPackages.find(packageName);
|
||||
if (it != installedPackages.end()) {
|
||||
|
1
getpkg/test_debug/debug-test
Executable file
1
getpkg/test_debug/debug-test
Executable file
@ -0,0 +1 @@
|
||||
#!/bin/bash\necho debug
|
1
getpkg/test_debug2/debug-test2
Executable file
1
getpkg/test_debug2/debug-test2
Executable file
@ -0,0 +1 @@
|
||||
#!/bin/bash\necho debug2
|
1
getpkg/test_upload.txt
Normal file
1
getpkg/test_upload.txt
Normal file
@ -0,0 +1 @@
|
||||
test content
|
Reference in New Issue
Block a user