forked from sergiotarxz/mangareader
Migrating to readmng, mangafox is fallen almost always and does to much to stop scrappers.
This commit is contained in:
parent
029c161519
commit
20fc7ff6a4
2
Makefile
2
Makefile
@ -6,4 +6,4 @@ LDFLAGS := $(shell pkg-config --libs ${LIBS})
|
|||||||
CC_COMMAND := ${CC} ${INCDIR} ${CFLAGS}
|
CC_COMMAND := ${CC} ${INCDIR} ${CFLAGS}
|
||||||
all: build
|
all: build
|
||||||
build:
|
build:
|
||||||
${CC_COMMAND} mangafox.c main.c -o main ${LDFLAGS} -ggdb
|
${CC_COMMAND} readmng.c manga.c main.c -o main ${LDFLAGS} -ggdb
|
||||||
|
@ -1,4 +1,62 @@
|
|||||||
|
#ifndef MANGA
|
||||||
|
#define MANGA
|
||||||
|
|
||||||
|
#include <libsoup/soup.h>
|
||||||
|
#include <libxml/HTMLparser.h>
|
||||||
|
#include <libxml/xpath.h>
|
||||||
|
#ifndef PCRE2_CODE_UNIT_WIDTH
|
||||||
|
#define PCRE2_CODE_UNIT_WIDTH 8
|
||||||
|
#include <pcre2.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define XML_COPY_NODE_RECURSIVE 2 | 1
|
||||||
|
|
||||||
struct Manga {
|
struct Manga {
|
||||||
char *title;
|
char *title;
|
||||||
char *image_url;
|
char *image_url;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct SplittedString {
|
||||||
|
struct String *substrings;
|
||||||
|
size_t n_strings;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct String {
|
||||||
|
char *content;
|
||||||
|
size_t size;
|
||||||
|
};
|
||||||
|
|
||||||
|
char *
|
||||||
|
get_request (const char *url, gsize *size_response_text);
|
||||||
|
xmlNodePtr *
|
||||||
|
find_class (xmlNodePtr node, char *class, size_t *len, xmlNodePtr *nodes,
|
||||||
|
int return_on_first);
|
||||||
|
void
|
||||||
|
print_debug_nodes (const xmlDocPtr html_document,
|
||||||
|
xmlNodePtr *nodes, size_t nodes_len);
|
||||||
|
char *
|
||||||
|
get_attr (xmlNodePtr const node, const char *attr_name);
|
||||||
|
void
|
||||||
|
copy_substring(const char *origin, char *dest, size_t dest_len, size_t start,
|
||||||
|
size_t len);
|
||||||
|
int
|
||||||
|
has_class (const char *class_attribute,
|
||||||
|
const char *class_to_check);
|
||||||
|
struct SplittedString *
|
||||||
|
split(char *re_str, size_t re_str_size, const char *subject, size_t subject_size);
|
||||||
|
char *
|
||||||
|
alloc_string(size_t len);
|
||||||
|
void
|
||||||
|
splitted_string_free (struct SplittedString *splitted_string);
|
||||||
|
void
|
||||||
|
iterate_string_to_split(struct SplittedString *splitted_string,
|
||||||
|
pcre2_code *re, int *will_break, const char *subject,
|
||||||
|
size_t subject_size, size_t *start_pos, size_t *offset);
|
||||||
|
xmlXPathObjectPtr
|
||||||
|
get_nodes_xpath_expression (const xmlDocPtr document, char *xpath);
|
||||||
|
xmlNodePtr *
|
||||||
|
loop_search_class (const xmlNodePtr node, xmlNodePtr *nodes,
|
||||||
|
const char * class, size_t *len);
|
||||||
|
char *
|
||||||
|
copy_binary_data (const char *input, size_t size);
|
||||||
|
#endif
|
||||||
|
@ -1,2 +0,0 @@
|
|||||||
void
|
|
||||||
retrieve_mangafox_title();
|
|
2
include/readmng.h
Normal file
2
include/readmng.h
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
struct Manga *
|
||||||
|
retrieve_readmng_title_mangas (size_t *const len);
|
141
main.c
141
main.c
@ -1,63 +1,132 @@
|
|||||||
#include <gtk/gtk.h>
|
#include <gtk/gtk.h>
|
||||||
#include <adwaita.h>
|
#include <adwaita.h>
|
||||||
|
|
||||||
#include <mangafox.h>
|
#include <readmng.h>
|
||||||
|
#include <manga.h>
|
||||||
|
|
||||||
AdwHeaderBar *
|
AdwHeaderBar *
|
||||||
create_headerbar (GtkBox *box);
|
create_headerbar (GtkBox *box);
|
||||||
GtkBox *
|
GtkBox *
|
||||||
create_main_box (AdwApplicationWindow *window);
|
create_main_box (AdwApplicationWindow *window);
|
||||||
|
GtkBox *
|
||||||
|
create_manga_container ();
|
||||||
|
AdwCarousel *
|
||||||
|
create_adw_caroulsel (GtkBox *box);
|
||||||
|
|
||||||
static void
|
void
|
||||||
|
fill_carousel_of_mangas (AdwCarousel *carousel);
|
||||||
|
|
||||||
|
void
|
||||||
activate (AdwApplication *app,
|
activate (AdwApplication *app,
|
||||||
gpointer user_data)
|
gpointer user_data)
|
||||||
{
|
{
|
||||||
GtkWidget *window =
|
GtkWidget *window =
|
||||||
adw_application_window_new (GTK_APPLICATION (app));
|
adw_application_window_new (GTK_APPLICATION (app));
|
||||||
GtkBox *box = create_main_box(
|
GtkBox *box = create_main_box(
|
||||||
ADW_APPLICATION_WINDOW
|
ADW_APPLICATION_WINDOW
|
||||||
(window));
|
(window));
|
||||||
create_headerbar (box);
|
AdwCarousel *carousel;
|
||||||
|
create_headerbar (box);
|
||||||
|
|
||||||
gtk_widget_show (window);
|
carousel = create_adw_caroulsel (box);
|
||||||
|
fill_carousel_of_mangas (carousel);
|
||||||
|
|
||||||
|
gtk_widget_show (window);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
fill_carousel_of_mangas (AdwCarousel *carousel) {
|
||||||
|
struct Manga *mangas;
|
||||||
|
struct Manga *manga;
|
||||||
|
GtkBox *manga_container;
|
||||||
|
size_t len_mangas = 0;
|
||||||
|
|
||||||
|
mangas = retrieve_readmng_title_mangas (&len_mangas);
|
||||||
|
for (int i = 0; i<len_mangas; i++) {
|
||||||
|
GtkWidget *picture;
|
||||||
|
GFileIOStream *iostream;
|
||||||
|
GFile *tmp_image;
|
||||||
|
GError *error = NULL;
|
||||||
|
|
||||||
|
size_t size_downloaded_image = 0;
|
||||||
|
char *downloaded_image;
|
||||||
|
|
||||||
|
manga = &mangas[i];
|
||||||
|
manga_container = create_manga_container (manga);
|
||||||
|
adw_carousel_append (carousel, GTK_WIDGET (manga_container));
|
||||||
|
downloaded_image = get_request (manga->image_url, &size_downloaded_image);
|
||||||
|
tmp_image = g_file_new_tmp ("mangareadertmpfileXXXXXX",
|
||||||
|
&iostream,
|
||||||
|
&error
|
||||||
|
);
|
||||||
|
if (error) {
|
||||||
|
fprintf (stderr, "Unable to read file: %s\n", error->message);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
error = NULL;
|
||||||
|
g_output_stream_write (g_io_stream_get_output_stream (G_IO_STREAM (iostream)),
|
||||||
|
downloaded_image, size_downloaded_image, NULL, &error);
|
||||||
|
if (error) {
|
||||||
|
fprintf (stderr, "Unable to write file: %s\n", error->message);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
picture = gtk_picture_new_for_file (tmp_image);
|
||||||
|
gtk_box_append (manga_container, picture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
GtkBox *
|
||||||
|
create_manga_container () {
|
||||||
|
GtkBox *manga_container;
|
||||||
|
manga_container = GTK_BOX (gtk_box_new(
|
||||||
|
GTK_ORIENTATION_HORIZONTAL,
|
||||||
|
0));
|
||||||
|
return manga_container;
|
||||||
}
|
}
|
||||||
|
|
||||||
GtkBox *
|
GtkBox *
|
||||||
create_main_box (AdwApplicationWindow *window) {
|
create_main_box (AdwApplicationWindow *window) {
|
||||||
GtkWidget *box = gtk_box_new(
|
GtkWidget *box = gtk_box_new(
|
||||||
GTK_ORIENTATION_VERTICAL,
|
GTK_ORIENTATION_VERTICAL,
|
||||||
10);
|
10);
|
||||||
adw_application_window_set_content(
|
adw_application_window_set_content(
|
||||||
window,
|
window,
|
||||||
box);
|
box);
|
||||||
return GTK_BOX (box);
|
return GTK_BOX (box);
|
||||||
|
}
|
||||||
|
|
||||||
|
AdwCarousel *
|
||||||
|
create_adw_caroulsel (GtkBox *box) {
|
||||||
|
GtkWidget *carousel = adw_carousel_new ();
|
||||||
|
gtk_box_append (box, carousel);
|
||||||
|
return ADW_CAROUSEL (carousel);
|
||||||
}
|
}
|
||||||
|
|
||||||
AdwHeaderBar *
|
AdwHeaderBar *
|
||||||
create_headerbar (GtkBox *box) {
|
create_headerbar (GtkBox *box) {
|
||||||
GtkWidget *title =
|
GtkWidget *title =
|
||||||
adw_window_title_new ("Window", NULL);
|
adw_window_title_new ("Window", NULL);
|
||||||
GtkWidget *header =
|
GtkWidget *header =
|
||||||
adw_header_bar_new();
|
adw_header_bar_new();
|
||||||
adw_header_bar_set_title_widget(
|
adw_header_bar_set_title_widget(
|
||||||
ADW_HEADER_BAR (header),
|
ADW_HEADER_BAR (header),
|
||||||
GTK_WIDGET (title));
|
GTK_WIDGET (title));
|
||||||
gtk_box_append (GTK_BOX (box), header);
|
gtk_box_append (box, header);
|
||||||
|
|
||||||
|
|
||||||
return ADW_HEADER_BAR (header);
|
return ADW_HEADER_BAR (header);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
main (int argc,
|
main (int argc,
|
||||||
char **argv)
|
char **argv)
|
||||||
{
|
{
|
||||||
AdwApplication *app;
|
AdwApplication *app;
|
||||||
retrieve_mangafox_title();
|
int status;
|
||||||
int status;
|
|
||||||
app = adw_application_new ("org.mangareader", G_APPLICATION_FLAGS_NONE);
|
app = adw_application_new ("org.mangareader", G_APPLICATION_FLAGS_NONE);
|
||||||
g_signal_connect (app, "activate", G_CALLBACK (activate), NULL);
|
g_signal_connect (app, "activate", G_CALLBACK (activate), NULL);
|
||||||
status = g_application_run (G_APPLICATION (app), argc, argv);
|
status = g_application_run (G_APPLICATION (app), argc, argv);
|
||||||
g_object_unref (app);
|
g_object_unref (app);
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
460
mangafox.c
460
mangafox.c
@ -1,460 +0,0 @@
|
|||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
#include <libsoup/soup.h>
|
|
||||||
#include <libxml/HTMLparser.h>
|
|
||||||
#include <libxml/xpath.h>
|
|
||||||
|
|
||||||
#ifndef PCRE2_CODE_UNIT_WIDTH
|
|
||||||
#define PCRE2_CODE_UNIT_WIDTH 8
|
|
||||||
#include <pcre2.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include <manga.h>
|
|
||||||
|
|
||||||
#define XML_COPY_NODE_RECURSIVE 2 | 1
|
|
||||||
|
|
||||||
const char *mangafox_url =
|
|
||||||
"https://mangafox.fun";
|
|
||||||
struct String {
|
|
||||||
char *content;
|
|
||||||
size_t size;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct SplittedString {
|
|
||||||
struct String *substrings;
|
|
||||||
size_t n_strings;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct Manga *
|
|
||||||
parse_main_mangafox_page (
|
|
||||||
const xmlDocPtr html_document,
|
|
||||||
const size_t *size);
|
|
||||||
xmlXPathObjectPtr
|
|
||||||
get_nodes_xpath_expression (
|
|
||||||
const xmlDocPtr document,
|
|
||||||
char *xpath);
|
|
||||||
char *
|
|
||||||
alloc_string(size_t len);
|
|
||||||
void
|
|
||||||
copy_substring(const char *origin, char *dest,
|
|
||||||
size_t dest_len, size_t start, size_t len);
|
|
||||||
void
|
|
||||||
print_classes (const char *class_attribute,
|
|
||||||
size_t class_attribute_size);
|
|
||||||
int
|
|
||||||
has_class (const char *class_attribute,
|
|
||||||
const char *class_to_check);
|
|
||||||
void
|
|
||||||
splitted_string_free (struct SplittedString *splitted_string);
|
|
||||||
|
|
||||||
struct SplittedString *
|
|
||||||
split(char *re_str, size_t re_str_size, const char *subject,
|
|
||||||
size_t subject_size);
|
|
||||||
void
|
|
||||||
iterate_string_to_split(struct SplittedString *splitted_string, pcre2_code *re, int *will_break, const char *subject,
|
|
||||||
size_t subject_size, size_t *start_pos, size_t *offset);
|
|
||||||
char *
|
|
||||||
get_request (const char *url, gsize *size_response_text);
|
|
||||||
xmlNodePtr *
|
|
||||||
loop_search_class(const xmlNodePtr node, xmlNodePtr *nodes,
|
|
||||||
const char * class, size_t *len);
|
|
||||||
void
|
|
||||||
print_debug_nodes (const xmlDocPtr html_document,
|
|
||||||
xmlNodePtr *nodes, size_t nodes_len);
|
|
||||||
xmlNodePtr *
|
|
||||||
find_all_manga_slide(const xmlDocPtr html_document,
|
|
||||||
size_t *len);
|
|
||||||
char *
|
|
||||||
get_attr (xmlNodePtr const node, const char *attr_name);
|
|
||||||
char *
|
|
||||||
get_manga_slide_cover(xmlNodePtr node);
|
|
||||||
char *
|
|
||||||
match_1 (char *re_str, char *subject);
|
|
||||||
xmlNodePtr
|
|
||||||
find_class(xmlNodePtr node, char *class);
|
|
||||||
char *
|
|
||||||
get_manga_slide_title(xmlNodePtr node);
|
|
||||||
|
|
||||||
void
|
|
||||||
retrieve_mangafox_title () {
|
|
||||||
xmlDocPtr html_response;
|
|
||||||
gsize size_response_text;
|
|
||||||
char *response_text = get_request (mangafox_url,
|
|
||||||
&size_response_text);
|
|
||||||
html_response = htmlReadMemory (response_text,
|
|
||||||
size_response_text,
|
|
||||||
NULL,
|
|
||||||
NULL,
|
|
||||||
HTML_PARSE_RECOVER | HTML_PARSE_NODEFDTD
|
|
||||||
| HTML_PARSE_NOERROR | HTML_PARSE_NOWARNING
|
|
||||||
);
|
|
||||||
size_t manga_size;
|
|
||||||
parse_main_mangafox_page (html_response, &manga_size);
|
|
||||||
xmlFreeDoc (html_response);
|
|
||||||
free (response_text);
|
|
||||||
}
|
|
||||||
|
|
||||||
char *
|
|
||||||
get_request (const char *url, gsize *size_response_text) {
|
|
||||||
SoupSession *soup_session;
|
|
||||||
SoupMessage *msg;
|
|
||||||
GValue response = G_VALUE_INIT;
|
|
||||||
guint status;
|
|
||||||
|
|
||||||
*size_response_text = 0;
|
|
||||||
|
|
||||||
g_value_init (&response, G_TYPE_BYTES);
|
|
||||||
|
|
||||||
soup_session = soup_session_new ();
|
|
||||||
msg = soup_message_new ("GET", url);
|
|
||||||
status = soup_session_send_message (soup_session, msg);
|
|
||||||
g_object_get_property(
|
|
||||||
G_OBJECT (msg),
|
|
||||||
"response-body-data",
|
|
||||||
&response);
|
|
||||||
|
|
||||||
printf("%u\n", status);
|
|
||||||
const char *html_response = g_bytes_get_data ((GBytes *)
|
|
||||||
g_value_peek_pointer (&response),
|
|
||||||
size_response_text);
|
|
||||||
|
|
||||||
char *return_value = g_strndup (html_response, *size_response_text);
|
|
||||||
|
|
||||||
g_value_unset (&response);
|
|
||||||
g_object_unref (soup_session);
|
|
||||||
g_object_unref (msg);
|
|
||||||
|
|
||||||
return return_value;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
struct Manga *
|
|
||||||
parse_main_mangafox_page (const xmlDocPtr html_document,
|
|
||||||
const size_t *size) {
|
|
||||||
xmlNodePtr *nodes;
|
|
||||||
xmlNodePtr node;
|
|
||||||
size_t nodes_len = 0;
|
|
||||||
|
|
||||||
nodes = find_all_manga_slide (html_document, &nodes_len);
|
|
||||||
print_debug_nodes (html_document, nodes, nodes_len);
|
|
||||||
for (int i = 0; i < nodes_len; i++) {
|
|
||||||
node = nodes[i];
|
|
||||||
char *cover = get_manga_slide_cover(node);
|
|
||||||
if (cover) {
|
|
||||||
printf ("%s\n", cover);
|
|
||||||
}
|
|
||||||
char *title = get_manga_slide_title (node);
|
|
||||||
if (title) {
|
|
||||||
printf ("%s\n", title);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for (int i = 0; i<nodes_len; i++) {
|
|
||||||
xmlNodePtr node = nodes[i];
|
|
||||||
xmlFreeNode (node);
|
|
||||||
}
|
|
||||||
g_free (nodes);
|
|
||||||
}
|
|
||||||
|
|
||||||
char *
|
|
||||||
get_manga_slide_title (xmlNodePtr node) {
|
|
||||||
xmlNodePtr m_slide_caption = find_class (node, "m-slide-caption");
|
|
||||||
if (!m_slide_caption) {
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
xmlNodePtr m_slide_title = find_class (m_slide_caption, "m-slide-title");
|
|
||||||
if (!m_slide_title) {
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
return (char *) xmlNodeGetContent (m_slide_title);
|
|
||||||
}
|
|
||||||
|
|
||||||
xmlNodePtr
|
|
||||||
find_class (xmlNodePtr node, char *class) {
|
|
||||||
for (xmlNodePtr child = node->children; child; child=child->next) {
|
|
||||||
char *attr = get_attr (child, "class");
|
|
||||||
if (attr && has_class (attr, class)) {
|
|
||||||
return child;
|
|
||||||
}
|
|
||||||
if (node->children) {
|
|
||||||
xmlNodePtr child = node->children;
|
|
||||||
for (;child;child=child->next) {
|
|
||||||
xmlNodePtr result = find_class (child, class);
|
|
||||||
if (result) {
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
char *
|
|
||||||
get_manga_slide_cover(xmlNodePtr node) {
|
|
||||||
for (xmlNodePtr child = node->children; child; child=child->next) {
|
|
||||||
char *attr = get_attr (child, "class");
|
|
||||||
if (attr && has_class (attr, "m-slide-background")) {
|
|
||||||
char *style = get_attr (child, "style");
|
|
||||||
char *match = match_1 ("background-image:url\\((.*?)\\)", style);
|
|
||||||
if (match) {
|
|
||||||
printf("%s\n", match);
|
|
||||||
return match;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
print_debug_nodes (const xmlDocPtr html_document,
|
|
||||||
xmlNodePtr *nodes, size_t nodes_len) {
|
|
||||||
xmlBufferPtr buffer = xmlBufferCreate ();
|
|
||||||
for (int i = 0; i < nodes_len; i++) {
|
|
||||||
xmlNodeDump (buffer, html_document, nodes[i],
|
|
||||||
0, 1);
|
|
||||||
}
|
|
||||||
xmlBufferDump (stdout, buffer);
|
|
||||||
xmlBufferFree (buffer);
|
|
||||||
}
|
|
||||||
|
|
||||||
xmlNodePtr *
|
|
||||||
find_all_manga_slide(const xmlDocPtr html_document,
|
|
||||||
size_t *len) {
|
|
||||||
xmlNodeSetPtr node_set;
|
|
||||||
xmlNodePtr *nodes;
|
|
||||||
xmlXPathObjectPtr xpath_result;
|
|
||||||
|
|
||||||
node_set = NULL;
|
|
||||||
nodes = NULL;
|
|
||||||
xpath_result = get_nodes_xpath_expression (html_document,
|
|
||||||
"//div[@class]");
|
|
||||||
|
|
||||||
if (!xpath_result) {
|
|
||||||
fprintf(stderr, "Empty xpath result\n");
|
|
||||||
goto cleanup_find_all_manga_slide;
|
|
||||||
}
|
|
||||||
node_set = xpath_result->nodesetval;
|
|
||||||
if (!node_set) {
|
|
||||||
fprintf(stderr, "No match\n");
|
|
||||||
goto cleanup_find_all_manga_slide;
|
|
||||||
}
|
|
||||||
for (int i = 0; i < node_set->nodeNr; i++) {
|
|
||||||
xmlNodePtr node = node_set->nodeTab[i];
|
|
||||||
nodes = loop_search_class (node, nodes, "manga-slide", len);
|
|
||||||
}
|
|
||||||
cleanup_find_all_manga_slide:
|
|
||||||
xmlXPathFreeObject (xpath_result);
|
|
||||||
|
|
||||||
return nodes;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
char *
|
|
||||||
get_attr (xmlNodePtr const node, const char *attr_name) {
|
|
||||||
char *return_value = NULL;
|
|
||||||
for (xmlAttr *attr = node->properties; attr; attr=attr->next) {
|
|
||||||
if (!xmlStrcmp(attr->name, (const xmlChar *) attr_name)
|
|
||||||
&& attr->children && attr->children->content) {
|
|
||||||
if (!attr->children->content) continue;
|
|
||||||
size_t content_len = strlen((char *)
|
|
||||||
attr->children->content);
|
|
||||||
return_value = alloc_string(content_len);
|
|
||||||
copy_substring ((char *) attr->children->content, return_value,
|
|
||||||
content_len,
|
|
||||||
0,
|
|
||||||
content_len);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return return_value;
|
|
||||||
}
|
|
||||||
|
|
||||||
xmlNodePtr *
|
|
||||||
loop_search_class (const xmlNodePtr node, xmlNodePtr *nodes,
|
|
||||||
const char * class, size_t *len) {
|
|
||||||
char *content = get_attr (node, "class");
|
|
||||||
if (!content) {
|
|
||||||
return nodes;
|
|
||||||
}
|
|
||||||
if (has_class (content, class)) {
|
|
||||||
(*len)++;
|
|
||||||
nodes = g_realloc (nodes, (sizeof *nodes) * *len);
|
|
||||||
nodes[(*len)-1] = xmlCopyNode(node, XML_COPY_NODE_RECURSIVE);
|
|
||||||
}
|
|
||||||
g_free (content);
|
|
||||||
return nodes;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
has_class (const char *class_attribute,
|
|
||||||
const char *class_to_check) {
|
|
||||||
char *re = "\\s+";
|
|
||||||
struct SplittedString *classes;
|
|
||||||
int return_value = 0;
|
|
||||||
classes = split(re, strlen(re), class_attribute,
|
|
||||||
strlen(class_attribute));
|
|
||||||
for (int i = 0; i<classes->n_strings; i++) {
|
|
||||||
if (strcmp(classes->substrings[i].content, class_to_check) == 0) {
|
|
||||||
return_value = 1;
|
|
||||||
goto cleanup_has_class;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
cleanup_has_class:
|
|
||||||
splitted_string_free (classes);
|
|
||||||
return return_value;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
splitted_string_free (struct SplittedString *splitted_string) {
|
|
||||||
for (int i = 0; i<splitted_string->n_strings; i++) {
|
|
||||||
g_free (splitted_string->substrings[i].content);
|
|
||||||
}
|
|
||||||
|
|
||||||
g_free (splitted_string->substrings);
|
|
||||||
g_free (splitted_string);
|
|
||||||
}
|
|
||||||
|
|
||||||
struct SplittedString *
|
|
||||||
split(char *re_str, size_t re_str_size, const char *subject, size_t subject_size) {
|
|
||||||
pcre2_code_8 *re;
|
|
||||||
size_t start_pos = 0;
|
|
||||||
size_t offset = 0;
|
|
||||||
int regex_compile_error;
|
|
||||||
PCRE2_SIZE error_offset;
|
|
||||||
struct SplittedString *splitted_string;
|
|
||||||
|
|
||||||
splitted_string = g_malloc (sizeof *splitted_string);
|
|
||||||
|
|
||||||
splitted_string->n_strings = 0;
|
|
||||||
splitted_string->substrings = NULL;
|
|
||||||
re = pcre2_compile ((PCRE2_SPTR8) re_str,
|
|
||||||
re_str_size, 0, ®ex_compile_error, &error_offset, NULL);
|
|
||||||
while (start_pos < subject_size) {
|
|
||||||
int will_break = 0;
|
|
||||||
iterate_string_to_split(splitted_string, re, &will_break,
|
|
||||||
subject, subject_size, &start_pos, &offset);
|
|
||||||
if (will_break) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pcre2_code_free (re);
|
|
||||||
re = NULL;
|
|
||||||
|
|
||||||
return splitted_string;
|
|
||||||
}
|
|
||||||
|
|
||||||
char *
|
|
||||||
match_1 (char *re_str, char *subject) {
|
|
||||||
pcre2_code *re;
|
|
||||||
pcre2_match_data *match_data;
|
|
||||||
|
|
||||||
char *return_value;
|
|
||||||
int regex_compile_error;
|
|
||||||
int rc;
|
|
||||||
size_t len_match = 0;
|
|
||||||
|
|
||||||
return_value = NULL;
|
|
||||||
PCRE2_SIZE error_offset;
|
|
||||||
|
|
||||||
re = pcre2_compile ((PCRE2_SPTR8) re_str, strlen (re_str), 0,
|
|
||||||
®ex_compile_error, &error_offset, NULL);
|
|
||||||
match_data = pcre2_match_data_create_from_pattern (re, NULL);
|
|
||||||
rc = pcre2_match (re, (PCRE2_SPTR8) subject, strlen (subject),
|
|
||||||
0, 0, match_data, NULL);
|
|
||||||
if (rc < 0 ) {
|
|
||||||
goto cleanup_match;
|
|
||||||
}
|
|
||||||
|
|
||||||
pcre2_substring_get_bynumber (match_data, 1, (PCRE2_UCHAR8**)
|
|
||||||
&return_value, &len_match);
|
|
||||||
cleanup_match:
|
|
||||||
pcre2_match_data_free (match_data);
|
|
||||||
pcre2_code_free (re);
|
|
||||||
return return_value;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
iterate_string_to_split(struct SplittedString *splitted_string, pcre2_code *re, int *will_break, const char *subject,
|
|
||||||
size_t subject_size, size_t *start_pos, size_t *offset) {
|
|
||||||
pcre2_match_data_8 *match_data;
|
|
||||||
PCRE2_SIZE *ovector;
|
|
||||||
int rc;
|
|
||||||
|
|
||||||
splitted_string->n_strings++;
|
|
||||||
match_data = pcre2_match_data_create_from_pattern_8 (re, NULL);
|
|
||||||
rc = pcre2_match ( re, (PCRE2_SPTR8) subject, subject_size, *start_pos, 0, match_data,
|
|
||||||
NULL);
|
|
||||||
if (splitted_string->substrings) {
|
|
||||||
splitted_string->substrings = g_realloc (splitted_string->substrings,
|
|
||||||
(sizeof *splitted_string->substrings) * (*offset + 1));
|
|
||||||
} else {
|
|
||||||
splitted_string->substrings = g_malloc (sizeof *splitted_string->substrings);
|
|
||||||
}
|
|
||||||
if (rc < 0) {
|
|
||||||
struct String *current_substring =
|
|
||||||
&splitted_string->substrings [*offset];
|
|
||||||
current_substring->content = alloc_string (subject_size
|
|
||||||
- *start_pos);
|
|
||||||
copy_substring (subject, current_substring->content,
|
|
||||||
subject_size,
|
|
||||||
*start_pos,
|
|
||||||
subject_size - *start_pos);
|
|
||||||
current_substring->size = subject_size - *start_pos;
|
|
||||||
|
|
||||||
*will_break = 1;
|
|
||||||
goto cleanup_iterate_string_to_split;
|
|
||||||
}
|
|
||||||
ovector = pcre2_get_ovector_pointer_8(match_data);
|
|
||||||
splitted_string->substrings[*offset].content = alloc_string (
|
|
||||||
ovector[0] - *start_pos);
|
|
||||||
copy_substring (subject, splitted_string->substrings[*offset]
|
|
||||||
.content,
|
|
||||||
subject_size,
|
|
||||||
*start_pos,
|
|
||||||
ovector[0] - *start_pos - 1);
|
|
||||||
splitted_string->substrings[*offset].size =
|
|
||||||
ovector[0] - *start_pos - 1;
|
|
||||||
|
|
||||||
*start_pos = ovector[1];
|
|
||||||
|
|
||||||
*offset += 1;
|
|
||||||
|
|
||||||
cleanup_iterate_string_to_split:
|
|
||||||
pcre2_match_data_free (match_data);
|
|
||||||
}
|
|
||||||
|
|
||||||
char *
|
|
||||||
alloc_string(size_t len) {
|
|
||||||
char * return_value = NULL;
|
|
||||||
return g_malloc (len + 1 * sizeof *return_value);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
copy_substring(const char *origin, char *dest, size_t dest_len, size_t start,
|
|
||||||
size_t len) {
|
|
||||||
size_t copying_offset = 0;
|
|
||||||
while (copying_offset < len) {
|
|
||||||
if (!(start+copying_offset <=dest_len)) {
|
|
||||||
fprintf(stderr, "Read attempt out of bounds.%ld %ld %ld\n", dest_len, start, len);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
dest[copying_offset] = origin[start+copying_offset];
|
|
||||||
copying_offset++;
|
|
||||||
}
|
|
||||||
dest[len] = '\0';
|
|
||||||
}
|
|
||||||
|
|
||||||
xmlXPathObjectPtr
|
|
||||||
get_nodes_xpath_expression (const xmlDocPtr document, char *xpath) {
|
|
||||||
xmlXPathContextPtr context;
|
|
||||||
xmlXPathObjectPtr result;
|
|
||||||
|
|
||||||
context = xmlXPathNewContext (document);
|
|
||||||
result = xmlXPathEvalExpression ((const xmlChar *)xpath, context);
|
|
||||||
|
|
||||||
xmlXPathFreeContext (context);
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
153
readmng.c
Normal file
153
readmng.c
Normal file
@ -0,0 +1,153 @@
|
|||||||
|
#include <manga.h>
|
||||||
|
#include <libxml/HTMLparser.h>
|
||||||
|
|
||||||
|
const char *readmng_url = "https://www.readmng.com/";
|
||||||
|
|
||||||
|
struct Manga *
|
||||||
|
parse_readmng_title_page (const xmlDocPtr html_document,
|
||||||
|
size_t *const len);
|
||||||
|
xmlNodePtr
|
||||||
|
retrieve_slides (const xmlDocPtr html_document);
|
||||||
|
xmlNodePtr
|
||||||
|
retrieve_ul_slides (xmlNodePtr const slides);
|
||||||
|
xmlNodePtr *
|
||||||
|
retrieve_li_slides (xmlNodePtr const slides, size_t *li_len);
|
||||||
|
xmlNodePtr
|
||||||
|
retrieve_img_from_thumnail (xmlNodePtr thumbnail);
|
||||||
|
xmlNodePtr
|
||||||
|
retrieve_thumbnail_from_li (xmlNodePtr current_li);
|
||||||
|
xmlNodePtr
|
||||||
|
retrieve_title_from_li (xmlNodePtr li);
|
||||||
|
struct Manga *
|
||||||
|
extract_manga_info_from_current_li (struct Manga *mangas,
|
||||||
|
xmlNodePtr current_li, size_t *len);
|
||||||
|
|
||||||
|
struct Manga *
|
||||||
|
retrieve_readmng_title_mangas (size_t *const len) {
|
||||||
|
xmlDocPtr html_response;
|
||||||
|
gsize size_response_text;
|
||||||
|
struct Manga *mangas;
|
||||||
|
char *response_text = get_request (readmng_url,
|
||||||
|
&size_response_text);
|
||||||
|
html_response = htmlReadMemory (response_text,
|
||||||
|
size_response_text,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
HTML_PARSE_RECOVER | HTML_PARSE_NODEFDTD
|
||||||
|
| HTML_PARSE_NOERROR | HTML_PARSE_NOWARNING
|
||||||
|
);
|
||||||
|
mangas = parse_readmng_title_page (html_response, len);
|
||||||
|
xmlFreeDoc (html_response);
|
||||||
|
g_free (response_text);
|
||||||
|
return mangas;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Manga *
|
||||||
|
parse_readmng_title_page (const xmlDocPtr html_document,
|
||||||
|
size_t *const len) {
|
||||||
|
struct Manga *mangas = NULL;
|
||||||
|
xmlNodePtr slides = retrieve_slides (html_document);
|
||||||
|
*len = 0;
|
||||||
|
size_t li_len = 0;
|
||||||
|
xmlNodePtr *li = retrieve_li_slides (slides, &li_len);
|
||||||
|
for (int i = 0; i<li_len; i++) {
|
||||||
|
xmlNodePtr current_li = li[i];
|
||||||
|
mangas = extract_manga_info_from_current_li (mangas, current_li,
|
||||||
|
len);
|
||||||
|
}
|
||||||
|
return mangas;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Manga *
|
||||||
|
extract_manga_info_from_current_li (struct Manga *mangas, xmlNodePtr current_li,
|
||||||
|
size_t *len) {
|
||||||
|
xmlNodePtr thumbnail = retrieve_thumbnail_from_li (current_li);
|
||||||
|
xmlNodePtr title = retrieve_title_from_li (current_li);
|
||||||
|
xmlNodePtr img;
|
||||||
|
|
||||||
|
if (thumbnail && title && (img = retrieve_img_from_thumnail (thumbnail))) {
|
||||||
|
(*len)++;
|
||||||
|
mangas = g_realloc(mangas, sizeof *mangas * *len);
|
||||||
|
struct Manga *manga = &mangas[*len-1];
|
||||||
|
manga->image_url = get_attr (img, "src");
|
||||||
|
manga->title = (char *) xmlNodeGetContent (title);
|
||||||
|
}
|
||||||
|
return mangas;
|
||||||
|
}
|
||||||
|
|
||||||
|
xmlNodePtr
|
||||||
|
retrieve_title_from_li (xmlNodePtr li) {
|
||||||
|
size_t title_len = 0;
|
||||||
|
xmlNodePtr *title = find_class (li, "title", &title_len, NULL, 1);
|
||||||
|
if (title_len) return title[0];
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
xmlNodePtr
|
||||||
|
retrieve_img_from_thumnail (xmlNodePtr thumbnail) {
|
||||||
|
for (xmlNodePtr child = thumbnail->children; child; child=child->next) {
|
||||||
|
if (!strcmp((char *)child->name, "img")) {
|
||||||
|
return child;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
xmlNodePtr
|
||||||
|
retrieve_thumbnail_from_li (xmlNodePtr current_li) {
|
||||||
|
size_t thumbnail_len = 0;
|
||||||
|
xmlNodePtr *thumbnail = find_class (current_li, "thumbnail",
|
||||||
|
&thumbnail_len, NULL, 1);
|
||||||
|
if (thumbnail_len) return thumbnail[0];
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
xmlNodePtr *
|
||||||
|
retrieve_li_slides (xmlNodePtr const slides, size_t *li_len) {
|
||||||
|
xmlNodePtr ul_slides = retrieve_ul_slides (slides);
|
||||||
|
xmlNodePtr *li = NULL;
|
||||||
|
for (xmlNodePtr child = ul_slides->children; child; child=child->next) {
|
||||||
|
(*li_len)++;
|
||||||
|
li = g_realloc(li, sizeof *li * *li_len);
|
||||||
|
li[*li_len-1] = xmlCopyNode(child, XML_COPY_NODE_RECURSIVE);
|
||||||
|
}
|
||||||
|
return li;
|
||||||
|
}
|
||||||
|
|
||||||
|
xmlNodePtr
|
||||||
|
retrieve_ul_slides (xmlNodePtr const slides) {
|
||||||
|
for (xmlNodePtr child = slides->children; child; child = child->next) {
|
||||||
|
if (!strcmp((char *) child->name, "ul")) {
|
||||||
|
return child;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
xmlNodePtr
|
||||||
|
retrieve_slides (const xmlDocPtr html_document) {
|
||||||
|
xmlNodePtr *nodes = NULL;
|
||||||
|
xmlXPathObjectPtr xpath_result = NULL;
|
||||||
|
xpath_result = get_nodes_xpath_expression (html_document,
|
||||||
|
"//div[@class]");
|
||||||
|
xmlNodePtr slides = NULL;
|
||||||
|
xmlNodeSetPtr node_set = NULL;
|
||||||
|
size_t matching_classes_len = 0;
|
||||||
|
|
||||||
|
node_set = xpath_result->nodesetval;
|
||||||
|
if (!node_set) {
|
||||||
|
fprintf(stderr, "No match\n");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
for (int i = 0; i < node_set->nodeNr; i++) {
|
||||||
|
xmlNodePtr node = node_set->nodeTab[i];
|
||||||
|
nodes = loop_search_class (node, nodes, "slides", &matching_classes_len);
|
||||||
|
}
|
||||||
|
if (nodes) {
|
||||||
|
slides = nodes[0];
|
||||||
|
}
|
||||||
|
if (xpath_result) {
|
||||||
|
xmlXPathFreeObject(xpath_result);
|
||||||
|
}
|
||||||
|
return slides;
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user