add komens attachments downloading functionality
Some checks are pending
/ sync-to-origin (push) Waiting to run
Some checks are pending
/ sync-to-origin (push) Waiting to run
This commit is contained in:
parent
1992b0e166
commit
a78f4536b2
@ -25,7 +25,7 @@ using nlohmann::json;
|
|||||||
|
|
||||||
std::vector<allocation> komens_allocated;
|
std::vector<allocation> komens_allocated;
|
||||||
|
|
||||||
void insert_content(WINDOW *content_win, WINDOW *attachment_win, size_t i,
|
void insert_content(WINDOW *content_win, WINDOW *attachment_win,
|
||||||
json &resp_from_api);
|
json &resp_from_api);
|
||||||
|
|
||||||
void komens_page(koment_type type) {
|
void komens_page(koment_type type) {
|
||||||
@ -145,8 +145,8 @@ void komens_page(koment_type type) {
|
|||||||
WINDOW *attachment_win = newwin(1, 1, LINES, COLS);
|
WINDOW *attachment_win = newwin(1, 1, LINES, COLS);
|
||||||
|
|
||||||
insert_content(content_win, attachment_win,
|
insert_content(content_win, attachment_win,
|
||||||
item_index(current_item(komens_choise_menu.menu)),
|
resp_from_api["Messages"][item_index(
|
||||||
resp_from_api);
|
current_item(komens_choise_menu.menu))]);
|
||||||
|
|
||||||
attron(COLOR_PAIR(COLOR_BLUE));
|
attron(COLOR_PAIR(COLOR_BLUE));
|
||||||
mvprintw(LINES - 2, 0,
|
mvprintw(LINES - 2, 0,
|
||||||
@ -169,22 +169,61 @@ void komens_page(koment_type type) {
|
|||||||
case 'k':
|
case 'k':
|
||||||
menu_driver(komens_choise_menu.menu, REQ_UP_ITEM);
|
menu_driver(komens_choise_menu.menu, REQ_UP_ITEM);
|
||||||
break;
|
break;
|
||||||
|
default:
|
||||||
|
if (c >= '1' && c <= '9') {
|
||||||
|
size_t index = c - '0' - 1;
|
||||||
|
if (index < resp_from_api["Messages"][item_index(
|
||||||
|
current_item(komens_choise_menu.menu))]["Attachments"]
|
||||||
|
.size()) {
|
||||||
|
|
||||||
|
std::string default_path =
|
||||||
|
"~/Downloads/" +
|
||||||
|
resp_from_api["Messages"][item_index(current_item(
|
||||||
|
komens_choise_menu.menu))]["Attachments"][index]["Name"]
|
||||||
|
.get<std::string>();
|
||||||
|
char path[256];
|
||||||
|
|
||||||
|
// Create input prompt at bottom of screen
|
||||||
|
move(LINES - 1, 0);
|
||||||
|
clrtoeol();
|
||||||
|
printw("Save path [%s]: ", default_path.c_str());
|
||||||
|
echo();
|
||||||
|
curs_set(1);
|
||||||
|
getnstr(path, sizeof(path) - 1);
|
||||||
|
if (strlen(path) == 0)
|
||||||
|
strcpy(path, default_path.c_str());
|
||||||
|
noecho();
|
||||||
|
curs_set(0);
|
||||||
|
move(LINES - 1, 0);
|
||||||
|
clrtoeol();
|
||||||
|
refresh();
|
||||||
|
|
||||||
|
// Download the attachment
|
||||||
|
bakaapi::download_attachment(
|
||||||
|
resp_from_api["Messages"][item_index(current_item(
|
||||||
|
komens_choise_menu.menu))]["Attachments"][index]["Id"]
|
||||||
|
.get<std::string>(),
|
||||||
|
path);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
insert_content(content_win, attachment_win,
|
insert_content(content_win, attachment_win,
|
||||||
item_index(current_item(komens_choise_menu.menu)),
|
resp_from_api["Messages"][item_index(
|
||||||
resp_from_api);
|
current_item(komens_choise_menu.menu))]);
|
||||||
wrefresh(komens_choise_menu.win);
|
wrefresh(komens_choise_menu.win);
|
||||||
}
|
}
|
||||||
delete_all(&komens_allocated);
|
delete_all(&komens_allocated);
|
||||||
}
|
}
|
||||||
|
|
||||||
void insert_content(WINDOW *content_win, WINDOW *attachment_win, size_t i,
|
void insert_content(WINDOW *content_win, WINDOW *attachment_win,
|
||||||
json &resp_from_api) {
|
json &message) {
|
||||||
wclear(content_win);
|
wclear(content_win);
|
||||||
mvwprintw(content_win, 0, 0, "%s",
|
mvwprintw(content_win, 0, 0, "%s",
|
||||||
html_to_string(resp_from_api.at("Messages")[i]["Text"]).c_str());
|
html_to_string(message.at("Text")).c_str());
|
||||||
wrefresh(content_win);
|
wrefresh(content_win);
|
||||||
if (!resp_from_api.at("Messages")[i]["Attachments"].empty()) {
|
if (!message.at("Attachments").empty()) {
|
||||||
|
|
||||||
size_t max_item_lenght = 0;
|
size_t max_item_lenght = 0;
|
||||||
{
|
{
|
||||||
@ -192,19 +231,15 @@ void insert_content(WINDOW *content_win, WINDOW *attachment_win, size_t i,
|
|||||||
size_t max_size_lenght = 0;
|
size_t max_size_lenght = 0;
|
||||||
size_t tmp_lenght;
|
size_t tmp_lenght;
|
||||||
|
|
||||||
for (size_t j = 0;
|
for (size_t j = 0; j < message.at("Attachments").size(); j++) {
|
||||||
j < resp_from_api.at("Messages")[i]["Attachments"].size(); j++) {
|
tmp_lenght =
|
||||||
tmp_lenght = resp_from_api.at("Messages")[i]["Attachments"][j]["Name"]
|
message.at("Attachments")[j]["Name"].get<std::string>().length();
|
||||||
.get<std::string>()
|
|
||||||
.length();
|
|
||||||
if (tmp_lenght > max_name_lenght) {
|
if (tmp_lenght > max_name_lenght) {
|
||||||
max_name_lenght = tmp_lenght;
|
max_name_lenght = tmp_lenght;
|
||||||
}
|
}
|
||||||
|
|
||||||
tmp_lenght =
|
tmp_lenght =
|
||||||
std::to_string(
|
std::to_string(message.at("Attachments")[j]["Size"].get<size_t>())
|
||||||
resp_from_api.at("Messages")[i]["Attachments"][j]["Size"]
|
|
||||||
.get<size_t>())
|
|
||||||
.length();
|
.length();
|
||||||
if (tmp_lenght > max_size_lenght) {
|
if (tmp_lenght > max_size_lenght) {
|
||||||
max_size_lenght = tmp_lenght;
|
max_size_lenght = tmp_lenght;
|
||||||
@ -221,17 +256,13 @@ void insert_content(WINDOW *content_win, WINDOW *attachment_win, size_t i,
|
|||||||
|
|
||||||
wborder(attachment_win, 0, ' ', 0, 0, 0, ACS_HLINE, 0, ACS_HLINE);
|
wborder(attachment_win, 0, ' ', 0, 0, 0, ACS_HLINE, 0, ACS_HLINE);
|
||||||
print_in_middle(attachment_win, 0, 0, max_item_lenght + 2, "Attachments",
|
print_in_middle(attachment_win, 0, 0, max_item_lenght + 2, "Attachments",
|
||||||
COLOR_RED);
|
COLOR_PAIR(COLOR_RED));
|
||||||
for (size_t j = 0;
|
for (size_t j = 0; j < message.at("Attachments").size(); j++) {
|
||||||
j < resp_from_api.at("Messages")[i]["Attachments"].size(); j++) {
|
|
||||||
|
|
||||||
mvwprintw(attachment_win, j + 1, 2, "%s %s",
|
mvwprintw(
|
||||||
resp_from_api.at("Messages")[i]["Attachments"][j]["Name"]
|
attachment_win, j + 1, 2, "%s %s",
|
||||||
.get<std::string>()
|
message.at("Attachments")[j]["Name"].get<std::string>().c_str(),
|
||||||
.c_str(),
|
std::to_string(message.at("Attachments")[j]["Size"].get<size_t>())
|
||||||
std::to_string(
|
|
||||||
resp_from_api.at("Messages")[i]["Attachments"][j]["Size"]
|
|
||||||
.get<size_t>())
|
|
||||||
.c_str());
|
.c_str());
|
||||||
|
|
||||||
wattron(attachment_win, COLOR_PAIR(COLOR_MAGENTA));
|
wattron(attachment_win, COLOR_PAIR(COLOR_MAGENTA));
|
||||||
|
29
src/main.cpp
29
src/main.cpp
@ -1,8 +1,10 @@
|
|||||||
#include "main.h"
|
#include "main.h"
|
||||||
#include "color.h"
|
#include "color.h"
|
||||||
|
#include "flags.h"
|
||||||
#include "helper_funcs.h"
|
#include "helper_funcs.h"
|
||||||
#include "main_menu.h"
|
#include "main_menu.h"
|
||||||
#include "net.h"
|
#include "net.h"
|
||||||
|
#include "types.h"
|
||||||
#include <csignal>
|
#include <csignal>
|
||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
#include <curl/curl.h>
|
#include <curl/curl.h>
|
||||||
@ -12,8 +14,6 @@
|
|||||||
#include <regex>
|
#include <regex>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#include "flags.h"
|
|
||||||
#include "types.h"
|
|
||||||
|
|
||||||
std::string baka_api_url;
|
std::string baka_api_url;
|
||||||
|
|
||||||
@ -37,12 +37,25 @@ int main(int argc, char **argv) {
|
|||||||
int opt;
|
int opt;
|
||||||
while ((opt = getopt(argc, argv, "hVvLS:")) != -1) {
|
while ((opt = getopt(argc, argv, "hVvLS:")) != -1) {
|
||||||
switch (opt) {
|
switch (opt) {
|
||||||
case 'h': PrintHelp(); break;
|
case 'h':
|
||||||
case 'V': PrintVersion(); break;
|
PrintHelp();
|
||||||
case 'v': config.verbose = true; break;
|
break;
|
||||||
case 'L': DeleteLogin(savedir_path); break;
|
case 'V':
|
||||||
case 'S': config.ignoressl = true; break;
|
PrintVersion();
|
||||||
default: std::cerr << RED"[ERROR]" << RESET" invalid option: " << (char)optopt << "\ntry: -h\n"; safe_exit(EINVAL);
|
break;
|
||||||
|
case 'v':
|
||||||
|
config.verbose = true;
|
||||||
|
break;
|
||||||
|
case 'L':
|
||||||
|
DeleteLogin(savedir_path);
|
||||||
|
break;
|
||||||
|
case 'S':
|
||||||
|
config.ignoressl = true;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
std::cerr << RED "[ERROR]" << RESET " invalid option: " << (char)optopt
|
||||||
|
<< "\ntry: -h\n";
|
||||||
|
safe_exit(EINVAL);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
105
src/net.cpp
105
src/net.cpp
@ -11,12 +11,17 @@
|
|||||||
#include <curl/curl.h>
|
#include <curl/curl.h>
|
||||||
#include <curses.h>
|
#include <curses.h>
|
||||||
#include <dirent.h>
|
#include <dirent.h>
|
||||||
|
#include <fcntl.h>
|
||||||
#include <format>
|
#include <format>
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <nlohmann/json.hpp>
|
#include <nlohmann/json.hpp>
|
||||||
#include <nlohmann/json_fwd.hpp>
|
#include <nlohmann/json_fwd.hpp>
|
||||||
|
#include <stdlib.h>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#define ROUND_UP(x, y) ((((x) + (y) - 1)) & ~((y) - 1))
|
||||||
|
|
||||||
using nlohmann::json;
|
using nlohmann::json;
|
||||||
|
|
||||||
@ -25,15 +30,44 @@ std::string access_token;
|
|||||||
CURL *curl = curl_easy_init();
|
CURL *curl = curl_easy_init();
|
||||||
|
|
||||||
// Callback function to write data into a std::string
|
// Callback function to write data into a std::string
|
||||||
size_t WriteCallback(void *contents, size_t size, size_t nmemb,
|
size_t WriteCallback_to_string(void *contents, size_t size, size_t nmemb,
|
||||||
std::string *userp) {
|
void *userp) {
|
||||||
size_t totalSize = size * nmemb;
|
size_t totalSize = size * nmemb;
|
||||||
userp->append((char *)contents, totalSize);
|
static_cast<std::string *>(userp)->append((char *)contents, totalSize);
|
||||||
return totalSize;
|
return totalSize;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::tuple<std::string, int> send_curl_request(std::string endpoint, metod type,
|
size_t WriteCallback_to_file(void *contents, size_t size, size_t nmemb,
|
||||||
std::string req_data) {
|
void *userp) {
|
||||||
|
const size_t sector_size = 4096;
|
||||||
|
size_t total = size * nmemb;
|
||||||
|
void *aligned_buf = aligned_alloc(sector_size, ROUND_UP(total, sector_size));
|
||||||
|
|
||||||
|
// Zero out the buffer before copying
|
||||||
|
memset(aligned_buf, 0, ROUND_UP(total, sector_size));
|
||||||
|
memcpy(aligned_buf, contents, total);
|
||||||
|
|
||||||
|
// Cast userp to fstream*
|
||||||
|
std::fstream *fileStream = static_cast<std::fstream *>(userp);
|
||||||
|
|
||||||
|
// Write to the file using fstream
|
||||||
|
fileStream->write(static_cast<char *>(aligned_buf), total);
|
||||||
|
|
||||||
|
// Check if write was successful
|
||||||
|
if (fileStream->fail()) {
|
||||||
|
free(aligned_buf);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
free(aligned_buf);
|
||||||
|
return total;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::tuple<std::string, int> send_curl_request(
|
||||||
|
std::string endpoint, metod type, std::string req_data,
|
||||||
|
size_t (*WriteCallback_function)(void *, size_t, size_t,
|
||||||
|
void *) = WriteCallback_to_string,
|
||||||
|
std::fstream *fileStream = nullptr) {
|
||||||
std::string response;
|
std::string response;
|
||||||
std::string url = baka_api_url + endpoint;
|
std::string url = baka_api_url + endpoint;
|
||||||
if (type == GET) {
|
if (type == GET) {
|
||||||
@ -42,15 +76,19 @@ std::tuple<std::string, int> send_curl_request(std::string endpoint, metod type,
|
|||||||
|
|
||||||
if (curl) {
|
if (curl) {
|
||||||
curl_easy_setopt(curl, CURLOPT_URL, url.c_str());
|
curl_easy_setopt(curl, CURLOPT_URL, url.c_str());
|
||||||
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, WriteCallback);
|
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, WriteCallback_function);
|
||||||
|
if (fileStream) {
|
||||||
|
curl_easy_setopt(curl, CURLOPT_WRITEDATA, fileStream);
|
||||||
|
} else {
|
||||||
curl_easy_setopt(curl, CURLOPT_WRITEDATA, &response);
|
curl_easy_setopt(curl, CURLOPT_WRITEDATA, &response);
|
||||||
|
}
|
||||||
if (config.ignoressl) {
|
if (config.ignoressl) {
|
||||||
curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L);
|
curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L);
|
||||||
curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, 0L);
|
curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, 0L);
|
||||||
}
|
}
|
||||||
|
|
||||||
curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1L);
|
curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1L);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_ACCEPT_ENCODING, "gzip, deflate");
|
||||||
|
|
||||||
struct curl_slist *headers = NULL;
|
struct curl_slist *headers = NULL;
|
||||||
headers = curl_slist_append(
|
headers = curl_slist_append(
|
||||||
@ -77,7 +115,15 @@ std::tuple<std::string, int> send_curl_request(std::string endpoint, metod type,
|
|||||||
std::cerr << RED "[ERROR] " << RESET "curl not initialised\n";
|
std::cerr << RED "[ERROR] " << RESET "curl not initialised\n";
|
||||||
safe_exit(20);
|
safe_exit(20);
|
||||||
}
|
}
|
||||||
curl_easy_perform(curl); // Perform the request
|
CURLcode curl_return_code = curl_easy_perform(curl);
|
||||||
|
if (curl_return_code != CURLE_OK) {
|
||||||
|
std::cerr << RED "[ERROR] " << RESET << "curl_easy_perform() failed: "
|
||||||
|
<< curl_easy_strerror(curl_return_code) << "\n";
|
||||||
|
safe_exit(21);
|
||||||
|
}
|
||||||
|
if (fileStream) {
|
||||||
|
fileStream->close();
|
||||||
|
}
|
||||||
|
|
||||||
int http_code = 0;
|
int http_code = 0;
|
||||||
curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &http_code);
|
curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &http_code);
|
||||||
@ -172,4 +218,47 @@ access_token_refreshed:
|
|||||||
|
|
||||||
return json::parse(response);
|
return json::parse(response);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int download_attachment(std::string id, std::string path) {
|
||||||
|
if (config.verbose) {
|
||||||
|
std::clog << "downloading attachment please wait...\n";
|
||||||
|
}
|
||||||
|
if (path[0] == '~') {
|
||||||
|
path = std::string(std::getenv("HOME")) + path.substr(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string savedir_path = std::filesystem::path(path).parent_path().string();
|
||||||
|
if (!std::filesystem::exists(savedir_path)) {
|
||||||
|
if (!std::filesystem::create_directories(savedir_path)) {
|
||||||
|
std::cerr << RED "[ERROR] " RESET
|
||||||
|
<< "Failed to create directory: " << savedir_path << "\n";
|
||||||
|
safe_exit(EXIT_FAILURE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::fstream fileStream(path, std::ios::out | std::ios::binary);
|
||||||
|
if (!fileStream.is_open()) {
|
||||||
|
std::cerr << RED "[ERROR] " RESET << "Failed to open file for writing\n";
|
||||||
|
safe_exit(EXIT_FAILURE);
|
||||||
|
}
|
||||||
|
|
||||||
|
is_access_token_empty();
|
||||||
|
access_token_refreshed:
|
||||||
|
std::string req_data =
|
||||||
|
std::format("Authorization=Bearer&access_token={}", access_token);
|
||||||
|
|
||||||
|
auto [response, http_code] =
|
||||||
|
send_curl_request(std::format("/api/3/komens/attachment/{}", id), GET,
|
||||||
|
req_data, WriteCallback_to_file, &fileStream);
|
||||||
|
if (http_code != 200) {
|
||||||
|
if (config.verbose) {
|
||||||
|
std::clog << BLUE "[LOG] " RESET << "download failed " << http_code
|
||||||
|
<< " is non 200 response\n";
|
||||||
|
}
|
||||||
|
refresh_access_token();
|
||||||
|
goto access_token_refreshed;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace bakaapi
|
} // namespace bakaapi
|
@ -19,5 +19,6 @@ void login(std::string username, std::string password);
|
|||||||
void refresh_access_token();
|
void refresh_access_token();
|
||||||
json get_data_from_endpoint(const std::string &endpoint, metod metod,
|
json get_data_from_endpoint(const std::string &endpoint, metod metod,
|
||||||
std::string additional_data = "");
|
std::string additional_data = "");
|
||||||
|
int download_attachment(std::string id, std::string path);
|
||||||
} // namespace bakaapi
|
} // namespace bakaapi
|
||||||
#endif
|
#endif
|
Loading…
x
Reference in New Issue
Block a user