0 /*
1 * ISC License
2 * Copyright (c) 2023 RMF <rawmonk@firemail.cc>
3 */
4 #ifdef __linux__
5 #define _DEFAULT_SOURCE
6 #include <syscall.h>
7 #endif
8 #include <unistd.h>
9 #include <pthread.h>
10 #include <stdio.h>
11 #include <stdlib.h>
12 #include <stdint.h>
13 #include <string.h>
14 #include "macro.h"
15 #include "error.h"
16 #include "sandbox.h"
17 #include "proc.h"
18 #include "url.h"
19 #include "page.h"
20 #include "gemini.h"
21 #include "request.h"
22 #include "strlcpy.h"
23 #include "strnstr.h"
24 #include "memory.h"
25 #define PARSER_INTERNAL
26 #include "parser.h"
27
28 struct parser {
29 int in;
30 int out;
31 pthread_mutex_t mutex;
32 };
33 struct parser request_parser;
34 struct parser page_parser;
35
36 void parser_request(int in, int out);
37
38 int parse_request(struct parser *parser, struct request *request) {
39
40 int ret, gemtext;
41 if (!parser) parser = &request_parser;
42
43 pthread_mutex_lock(&parser->mutex);
44
45 write(parser->out, P(request->length));
46 write(parser->out, request->data, request->length);
47
48 if ((ret = vread(parser->in, P(request->status)))) goto fail;
49 if (request->status == -1) {
50 int error;
51 if ((ret = vread(parser->in, P(error)))) goto fail;
52 ret = error;
53 goto fail;
54 }
55 if ((ret = vread(parser->in, V(request->meta)))) goto fail;
56 if ((ret = vread(parser->in, P(request->page.mime)))) goto fail;
57 if ((ret = vread(parser->in, P(request->page.offset)))) goto fail;
58
59 gemtext = request->status == GMI_SUCCESS &&
60 is_gemtext(V(request->meta));
61 request->page.links_count = 0;
62 request->page.links = NULL;
63 ret = 0;
64 while (gemtext) {
65
66 size_t length = 0;
67 char url[MAX_URL] = {0};
68 void *tmp;
69 char *ptr;
70
71 if (vread(parser->in, P(length))) {
72 ret = -1;
73 break;
74 }
75 if (length == (size_t)-1) break;
76 if (!length || vread(parser->in, url, length)) {
77 ret = -1;
78 break;
79 }
80
81 length++;
82 if ((ret = readonly(url, length, &ptr))) break;
83
84 tmp = realloc(request->page.links,
85 sizeof(char*) * (request->page.links_count + 1));
86 if (!tmp) {
87 ret = ERROR_MEMORY_FAILURE;
88 break;
89 }
90 request->page.links = tmp;
91 request->page.links[request->page.links_count] = ptr;
92 request->page.links_count++;
93 }
94
95 if (gemtext && (ret = vread(parser->in, V(request->page.title))))
96 goto fail;
97 if (!gemtext || !request->page.title[0])
98 STRLCPY(request->page.title, request->url);
99 fail:
100 pthread_mutex_unlock(&parser->mutex);
101 return ret;
102 }
103
104 void parser_page(int in, int out) {
105 char *data = NULL;
106 parser_sandbox(out, "vgmi [page]");
107 while (1) { /* TODO: send error code */
108
109 int ret;
110 size_t length;
111 int width;
112 int mime;
113
114 if (vread(in, &length, sizeof(length))) break;
115 if (vread(in, &width, sizeof(width))) break;
116 if (vread(in, &mime, sizeof(mime))) break;
117
118 switch (mime) {
119 case MIME_GEMTEXT:
120 ret = parse_gemtext(in, length, width, out);
121 break;
122 case MIME_PLAIN:
123 ret = parse_plain(in, length, width, out);
124 break;
125 default:
126 ret = parse_binary(in, length, width, out);
127 break;
128 }
129
130 if (ret) break;
131
132 free(data);
133 data = NULL;
134 }
135 free(data);
136 }
137
138 int parse_page(struct parser *parser, struct request *request, int width) {
139
140 int ret;
141 size_t length;
142 if (!parser) parser = &page_parser;
143
144 pthread_mutex_lock(&parser->mutex);
145
146 length = request->length - request->page.offset;
147 write(parser->out, P(length));
148 write(parser->out, P(width));
149 write(parser->out, P(request->page.mime));
150
151 request->page.width = width;
152 ret = page_update(parser->in, parser->out,
153 &request->data[request->page.offset],
154 length, &request->page);
155
156 pthread_mutex_unlock(&parser->mutex);
157 return ret;
158 }
159
160 int parser_create(struct parser *parser, int type) {
161 switch (type) {
162 case PARSER_GEMTEXT:
163 proc_fork("--page", &parser->in, &parser->out);
164 break;
165 case PARSER_REQUEST:
166 proc_fork("--request", &parser->in, &parser->out);
167 break;
168 }
169 return 0;
170 }
171
172 int parser_request_create() {
173 return parser_create(&request_parser, PARSER_REQUEST);
174 }
175
176 int parser_page_create() {
177 return parser_create(&page_parser, PARSER_GEMTEXT);
178 }
179
180 int parser_sandbox(int out, const char *title) {
181 uint8_t byte;
182 sandbox_set_name(title);
183 if (sandbox_isolate()) return -1;
184 byte = 0;
185 write(out, &byte, 1);
186 return 0;
187 }
188